message-compiler.esm-browser.js 58 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625
  1. /*!
  2. * message-compiler v9.13.1
  3. * (c) 2024 kazuya kawaguchi
  4. * Released under the MIT License.
  5. */
  6. const LOCATION_STUB = {
  7. start: { line: 1, column: 1, offset: 0 },
  8. end: { line: 1, column: 1, offset: 0 }
  9. };
  10. function createPosition(line, column, offset) {
  11. return { line, column, offset };
  12. }
  13. function createLocation(start, end, source) {
  14. const loc = { start, end };
  15. if (source != null) {
  16. loc.source = source;
  17. }
  18. return loc;
  19. }
  20. /**
  21. * Original Utilities
  22. * written by kazuya kawaguchi
  23. */
  24. const RE_ARGS = /\{([0-9a-zA-Z]+)\}/g;
  25. /* eslint-disable */
  26. function format(message, ...args) {
  27. if (args.length === 1 && isObject(args[0])) {
  28. args = args[0];
  29. }
  30. if (!args || !args.hasOwnProperty) {
  31. args = {};
  32. }
  33. return message.replace(RE_ARGS, (match, identifier) => {
  34. return args.hasOwnProperty(identifier) ? args[identifier] : '';
  35. });
  36. }
  37. const assign = Object.assign;
  38. const isString = (val) => typeof val === 'string';
  39. // eslint-disable-next-line @typescript-eslint/no-explicit-any
  40. const isObject = (val) => val !== null && typeof val === 'object';
  41. function join(items, separator = '') {
  42. return items.reduce((str, item, index) => (index === 0 ? str + item : str + separator + item), '');
  43. }
  44. const CompileWarnCodes = {
  45. USE_MODULO_SYNTAX: 1,
  46. __EXTEND_POINT__: 2
  47. };
  48. /** @internal */
  49. const warnMessages = {
  50. [CompileWarnCodes.USE_MODULO_SYNTAX]: `Use modulo before '{{0}}'.`
  51. };
  52. function createCompileWarn(code, loc, ...args) {
  53. const msg = format(warnMessages[code] || '', ...(args || [])) ;
  54. const message = { message: String(msg), code };
  55. if (loc) {
  56. message.location = loc;
  57. }
  58. return message;
  59. }
  60. const CompileErrorCodes = {
  61. // tokenizer error codes
  62. EXPECTED_TOKEN: 1,
  63. INVALID_TOKEN_IN_PLACEHOLDER: 2,
  64. UNTERMINATED_SINGLE_QUOTE_IN_PLACEHOLDER: 3,
  65. UNKNOWN_ESCAPE_SEQUENCE: 4,
  66. INVALID_UNICODE_ESCAPE_SEQUENCE: 5,
  67. UNBALANCED_CLOSING_BRACE: 6,
  68. UNTERMINATED_CLOSING_BRACE: 7,
  69. EMPTY_PLACEHOLDER: 8,
  70. NOT_ALLOW_NEST_PLACEHOLDER: 9,
  71. INVALID_LINKED_FORMAT: 10,
  72. // parser error codes
  73. MUST_HAVE_MESSAGES_IN_PLURAL: 11,
  74. UNEXPECTED_EMPTY_LINKED_MODIFIER: 12,
  75. UNEXPECTED_EMPTY_LINKED_KEY: 13,
  76. UNEXPECTED_LEXICAL_ANALYSIS: 14,
  77. // generator error codes
  78. UNHANDLED_CODEGEN_NODE_TYPE: 15,
  79. // minifier error codes
  80. UNHANDLED_MINIFIER_NODE_TYPE: 16,
  81. // Special value for higher-order compilers to pick up the last code
  82. // to avoid collision of error codes. This should always be kept as the last
  83. // item.
  84. __EXTEND_POINT__: 17
  85. };
  86. /** @internal */
  87. const errorMessages = {
  88. // tokenizer error messages
  89. [CompileErrorCodes.EXPECTED_TOKEN]: `Expected token: '{0}'`,
  90. [CompileErrorCodes.INVALID_TOKEN_IN_PLACEHOLDER]: `Invalid token in placeholder: '{0}'`,
  91. [CompileErrorCodes.UNTERMINATED_SINGLE_QUOTE_IN_PLACEHOLDER]: `Unterminated single quote in placeholder`,
  92. [CompileErrorCodes.UNKNOWN_ESCAPE_SEQUENCE]: `Unknown escape sequence: \\{0}`,
  93. [CompileErrorCodes.INVALID_UNICODE_ESCAPE_SEQUENCE]: `Invalid unicode escape sequence: {0}`,
  94. [CompileErrorCodes.UNBALANCED_CLOSING_BRACE]: `Unbalanced closing brace`,
  95. [CompileErrorCodes.UNTERMINATED_CLOSING_BRACE]: `Unterminated closing brace`,
  96. [CompileErrorCodes.EMPTY_PLACEHOLDER]: `Empty placeholder`,
  97. [CompileErrorCodes.NOT_ALLOW_NEST_PLACEHOLDER]: `Not allowed nest placeholder`,
  98. [CompileErrorCodes.INVALID_LINKED_FORMAT]: `Invalid linked format`,
  99. // parser error messages
  100. [CompileErrorCodes.MUST_HAVE_MESSAGES_IN_PLURAL]: `Plural must have messages`,
  101. [CompileErrorCodes.UNEXPECTED_EMPTY_LINKED_MODIFIER]: `Unexpected empty linked modifier`,
  102. [CompileErrorCodes.UNEXPECTED_EMPTY_LINKED_KEY]: `Unexpected empty linked key`,
  103. [CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS]: `Unexpected lexical analysis in token: '{0}'`,
  104. // generator error messages
  105. [CompileErrorCodes.UNHANDLED_CODEGEN_NODE_TYPE]: `unhandled codegen node type: '{0}'`,
  106. // minimizer error messages
  107. [CompileErrorCodes.UNHANDLED_MINIFIER_NODE_TYPE]: `unhandled mimifier node type: '{0}'`
  108. };
  109. function createCompileError(code, loc, options = {}) {
  110. const { domain, messages, args } = options;
  111. const msg = format((messages || errorMessages)[code] || '', ...(args || []))
  112. ;
  113. const error = new SyntaxError(String(msg));
  114. error.code = code;
  115. if (loc) {
  116. error.location = loc;
  117. }
  118. error.domain = domain;
  119. return error;
  120. }
  121. /** @internal */
  122. function defaultOnError(error) {
  123. throw error;
  124. }
  125. // eslint-disable-next-line no-useless-escape
  126. const RE_HTML_TAG = /<\/?[\w\s="/.':;#-\/]+>/;
  127. const detectHtmlTag = (source) => RE_HTML_TAG.test(source);
  128. const CHAR_SP = ' ';
  129. const CHAR_CR = '\r';
  130. const CHAR_LF = '\n';
  131. const CHAR_LS = String.fromCharCode(0x2028);
  132. const CHAR_PS = String.fromCharCode(0x2029);
  133. function createScanner(str) {
  134. const _buf = str;
  135. let _index = 0;
  136. let _line = 1;
  137. let _column = 1;
  138. let _peekOffset = 0;
  139. const isCRLF = (index) => _buf[index] === CHAR_CR && _buf[index + 1] === CHAR_LF;
  140. const isLF = (index) => _buf[index] === CHAR_LF;
  141. const isPS = (index) => _buf[index] === CHAR_PS;
  142. const isLS = (index) => _buf[index] === CHAR_LS;
  143. const isLineEnd = (index) => isCRLF(index) || isLF(index) || isPS(index) || isLS(index);
  144. const index = () => _index;
  145. const line = () => _line;
  146. const column = () => _column;
  147. const peekOffset = () => _peekOffset;
  148. const charAt = (offset) => isCRLF(offset) || isPS(offset) || isLS(offset) ? CHAR_LF : _buf[offset];
  149. const currentChar = () => charAt(_index);
  150. const currentPeek = () => charAt(_index + _peekOffset);
  151. function next() {
  152. _peekOffset = 0;
  153. if (isLineEnd(_index)) {
  154. _line++;
  155. _column = 0;
  156. }
  157. if (isCRLF(_index)) {
  158. _index++;
  159. }
  160. _index++;
  161. _column++;
  162. return _buf[_index];
  163. }
  164. function peek() {
  165. if (isCRLF(_index + _peekOffset)) {
  166. _peekOffset++;
  167. }
  168. _peekOffset++;
  169. return _buf[_index + _peekOffset];
  170. }
  171. function reset() {
  172. _index = 0;
  173. _line = 1;
  174. _column = 1;
  175. _peekOffset = 0;
  176. }
  177. function resetPeek(offset = 0) {
  178. _peekOffset = offset;
  179. }
  180. function skipToPeek() {
  181. const target = _index + _peekOffset;
  182. // eslint-disable-next-line no-unmodified-loop-condition
  183. while (target !== _index) {
  184. next();
  185. }
  186. _peekOffset = 0;
  187. }
  188. return {
  189. index,
  190. line,
  191. column,
  192. peekOffset,
  193. charAt,
  194. currentChar,
  195. currentPeek,
  196. next,
  197. peek,
  198. reset,
  199. resetPeek,
  200. skipToPeek
  201. };
  202. }
  203. const EOF = undefined;
  204. const DOT = '.';
  205. const LITERAL_DELIMITER = "'";
  206. const ERROR_DOMAIN$3 = 'tokenizer';
  207. function createTokenizer(source, options = {}) {
  208. const location = options.location !== false;
  209. const _scnr = createScanner(source);
  210. const currentOffset = () => _scnr.index();
  211. const currentPosition = () => createPosition(_scnr.line(), _scnr.column(), _scnr.index());
  212. const _initLoc = currentPosition();
  213. const _initOffset = currentOffset();
  214. const _context = {
  215. currentType: 14 /* TokenTypes.EOF */,
  216. offset: _initOffset,
  217. startLoc: _initLoc,
  218. endLoc: _initLoc,
  219. lastType: 14 /* TokenTypes.EOF */,
  220. lastOffset: _initOffset,
  221. lastStartLoc: _initLoc,
  222. lastEndLoc: _initLoc,
  223. braceNest: 0,
  224. inLinked: false,
  225. text: ''
  226. };
  227. const context = () => _context;
  228. const { onError } = options;
  229. function emitError(code, pos, offset, ...args) {
  230. const ctx = context();
  231. pos.column += offset;
  232. pos.offset += offset;
  233. if (onError) {
  234. const loc = location ? createLocation(ctx.startLoc, pos) : null;
  235. const err = createCompileError(code, loc, {
  236. domain: ERROR_DOMAIN$3,
  237. args
  238. });
  239. onError(err);
  240. }
  241. }
  242. function getToken(context, type, value) {
  243. context.endLoc = currentPosition();
  244. context.currentType = type;
  245. const token = { type };
  246. if (location) {
  247. token.loc = createLocation(context.startLoc, context.endLoc);
  248. }
  249. if (value != null) {
  250. token.value = value;
  251. }
  252. return token;
  253. }
  254. const getEndToken = (context) => getToken(context, 14 /* TokenTypes.EOF */);
  255. function eat(scnr, ch) {
  256. if (scnr.currentChar() === ch) {
  257. scnr.next();
  258. return ch;
  259. }
  260. else {
  261. emitError(CompileErrorCodes.EXPECTED_TOKEN, currentPosition(), 0, ch);
  262. return '';
  263. }
  264. }
  265. function peekSpaces(scnr) {
  266. let buf = '';
  267. while (scnr.currentPeek() === CHAR_SP || scnr.currentPeek() === CHAR_LF) {
  268. buf += scnr.currentPeek();
  269. scnr.peek();
  270. }
  271. return buf;
  272. }
  273. function skipSpaces(scnr) {
  274. const buf = peekSpaces(scnr);
  275. scnr.skipToPeek();
  276. return buf;
  277. }
  278. function isIdentifierStart(ch) {
  279. if (ch === EOF) {
  280. return false;
  281. }
  282. const cc = ch.charCodeAt(0);
  283. return ((cc >= 97 && cc <= 122) || // a-z
  284. (cc >= 65 && cc <= 90) || // A-Z
  285. cc === 95 // _
  286. );
  287. }
  288. function isNumberStart(ch) {
  289. if (ch === EOF) {
  290. return false;
  291. }
  292. const cc = ch.charCodeAt(0);
  293. return cc >= 48 && cc <= 57; // 0-9
  294. }
  295. function isNamedIdentifierStart(scnr, context) {
  296. const { currentType } = context;
  297. if (currentType !== 2 /* TokenTypes.BraceLeft */) {
  298. return false;
  299. }
  300. peekSpaces(scnr);
  301. const ret = isIdentifierStart(scnr.currentPeek());
  302. scnr.resetPeek();
  303. return ret;
  304. }
  305. function isListIdentifierStart(scnr, context) {
  306. const { currentType } = context;
  307. if (currentType !== 2 /* TokenTypes.BraceLeft */) {
  308. return false;
  309. }
  310. peekSpaces(scnr);
  311. const ch = scnr.currentPeek() === '-' ? scnr.peek() : scnr.currentPeek();
  312. const ret = isNumberStart(ch);
  313. scnr.resetPeek();
  314. return ret;
  315. }
  316. function isLiteralStart(scnr, context) {
  317. const { currentType } = context;
  318. if (currentType !== 2 /* TokenTypes.BraceLeft */) {
  319. return false;
  320. }
  321. peekSpaces(scnr);
  322. const ret = scnr.currentPeek() === LITERAL_DELIMITER;
  323. scnr.resetPeek();
  324. return ret;
  325. }
  326. function isLinkedDotStart(scnr, context) {
  327. const { currentType } = context;
  328. if (currentType !== 8 /* TokenTypes.LinkedAlias */) {
  329. return false;
  330. }
  331. peekSpaces(scnr);
  332. const ret = scnr.currentPeek() === "." /* TokenChars.LinkedDot */;
  333. scnr.resetPeek();
  334. return ret;
  335. }
  336. function isLinkedModifierStart(scnr, context) {
  337. const { currentType } = context;
  338. if (currentType !== 9 /* TokenTypes.LinkedDot */) {
  339. return false;
  340. }
  341. peekSpaces(scnr);
  342. const ret = isIdentifierStart(scnr.currentPeek());
  343. scnr.resetPeek();
  344. return ret;
  345. }
  346. function isLinkedDelimiterStart(scnr, context) {
  347. const { currentType } = context;
  348. if (!(currentType === 8 /* TokenTypes.LinkedAlias */ ||
  349. currentType === 12 /* TokenTypes.LinkedModifier */)) {
  350. return false;
  351. }
  352. peekSpaces(scnr);
  353. const ret = scnr.currentPeek() === ":" /* TokenChars.LinkedDelimiter */;
  354. scnr.resetPeek();
  355. return ret;
  356. }
  357. function isLinkedReferStart(scnr, context) {
  358. const { currentType } = context;
  359. if (currentType !== 10 /* TokenTypes.LinkedDelimiter */) {
  360. return false;
  361. }
  362. const fn = () => {
  363. const ch = scnr.currentPeek();
  364. if (ch === "{" /* TokenChars.BraceLeft */) {
  365. return isIdentifierStart(scnr.peek());
  366. }
  367. else if (ch === "@" /* TokenChars.LinkedAlias */ ||
  368. ch === "%" /* TokenChars.Modulo */ ||
  369. ch === "|" /* TokenChars.Pipe */ ||
  370. ch === ":" /* TokenChars.LinkedDelimiter */ ||
  371. ch === "." /* TokenChars.LinkedDot */ ||
  372. ch === CHAR_SP ||
  373. !ch) {
  374. return false;
  375. }
  376. else if (ch === CHAR_LF) {
  377. scnr.peek();
  378. return fn();
  379. }
  380. else {
  381. // other characters
  382. return isTextStart(scnr, false);
  383. }
  384. };
  385. const ret = fn();
  386. scnr.resetPeek();
  387. return ret;
  388. }
  389. function isPluralStart(scnr) {
  390. peekSpaces(scnr);
  391. const ret = scnr.currentPeek() === "|" /* TokenChars.Pipe */;
  392. scnr.resetPeek();
  393. return ret;
  394. }
  395. function detectModuloStart(scnr) {
  396. const spaces = peekSpaces(scnr);
  397. const ret = scnr.currentPeek() === "%" /* TokenChars.Modulo */ &&
  398. scnr.peek() === "{" /* TokenChars.BraceLeft */;
  399. scnr.resetPeek();
  400. return {
  401. isModulo: ret,
  402. hasSpace: spaces.length > 0
  403. };
  404. }
  405. function isTextStart(scnr, reset = true) {
  406. const fn = (hasSpace = false, prev = '', detectModulo = false) => {
  407. const ch = scnr.currentPeek();
  408. if (ch === "{" /* TokenChars.BraceLeft */) {
  409. return prev === "%" /* TokenChars.Modulo */ ? false : hasSpace;
  410. }
  411. else if (ch === "@" /* TokenChars.LinkedAlias */ || !ch) {
  412. return prev === "%" /* TokenChars.Modulo */ ? true : hasSpace;
  413. }
  414. else if (ch === "%" /* TokenChars.Modulo */) {
  415. scnr.peek();
  416. return fn(hasSpace, "%" /* TokenChars.Modulo */, true);
  417. }
  418. else if (ch === "|" /* TokenChars.Pipe */) {
  419. return prev === "%" /* TokenChars.Modulo */ || detectModulo
  420. ? true
  421. : !(prev === CHAR_SP || prev === CHAR_LF);
  422. }
  423. else if (ch === CHAR_SP) {
  424. scnr.peek();
  425. return fn(true, CHAR_SP, detectModulo);
  426. }
  427. else if (ch === CHAR_LF) {
  428. scnr.peek();
  429. return fn(true, CHAR_LF, detectModulo);
  430. }
  431. else {
  432. return true;
  433. }
  434. };
  435. const ret = fn();
  436. reset && scnr.resetPeek();
  437. return ret;
  438. }
  439. function takeChar(scnr, fn) {
  440. const ch = scnr.currentChar();
  441. if (ch === EOF) {
  442. return EOF;
  443. }
  444. if (fn(ch)) {
  445. scnr.next();
  446. return ch;
  447. }
  448. return null;
  449. }
  450. function isIdentifier(ch) {
  451. const cc = ch.charCodeAt(0);
  452. return ((cc >= 97 && cc <= 122) || // a-z
  453. (cc >= 65 && cc <= 90) || // A-Z
  454. (cc >= 48 && cc <= 57) || // 0-9
  455. cc === 95 || // _
  456. cc === 36 // $
  457. );
  458. }
  459. function takeIdentifierChar(scnr) {
  460. return takeChar(scnr, isIdentifier);
  461. }
  462. function isNamedIdentifier(ch) {
  463. const cc = ch.charCodeAt(0);
  464. return ((cc >= 97 && cc <= 122) || // a-z
  465. (cc >= 65 && cc <= 90) || // A-Z
  466. (cc >= 48 && cc <= 57) || // 0-9
  467. cc === 95 || // _
  468. cc === 36 || // $
  469. cc === 45 // -
  470. );
  471. }
  472. function takeNamedIdentifierChar(scnr) {
  473. return takeChar(scnr, isNamedIdentifier);
  474. }
  475. function isDigit(ch) {
  476. const cc = ch.charCodeAt(0);
  477. return cc >= 48 && cc <= 57; // 0-9
  478. }
  479. function takeDigit(scnr) {
  480. return takeChar(scnr, isDigit);
  481. }
  482. function isHexDigit(ch) {
  483. const cc = ch.charCodeAt(0);
  484. return ((cc >= 48 && cc <= 57) || // 0-9
  485. (cc >= 65 && cc <= 70) || // A-F
  486. (cc >= 97 && cc <= 102)); // a-f
  487. }
  488. function takeHexDigit(scnr) {
  489. return takeChar(scnr, isHexDigit);
  490. }
  491. function getDigits(scnr) {
  492. let ch = '';
  493. let num = '';
  494. while ((ch = takeDigit(scnr))) {
  495. num += ch;
  496. }
  497. return num;
  498. }
  499. function readModulo(scnr) {
  500. skipSpaces(scnr);
  501. const ch = scnr.currentChar();
  502. if (ch !== "%" /* TokenChars.Modulo */) {
  503. emitError(CompileErrorCodes.EXPECTED_TOKEN, currentPosition(), 0, ch);
  504. }
  505. scnr.next();
  506. return "%" /* TokenChars.Modulo */;
  507. }
  508. function readText(scnr) {
  509. let buf = '';
  510. // eslint-disable-next-line no-constant-condition
  511. while (true) {
  512. const ch = scnr.currentChar();
  513. if (ch === "{" /* TokenChars.BraceLeft */ ||
  514. ch === "}" /* TokenChars.BraceRight */ ||
  515. ch === "@" /* TokenChars.LinkedAlias */ ||
  516. ch === "|" /* TokenChars.Pipe */ ||
  517. !ch) {
  518. break;
  519. }
  520. else if (ch === "%" /* TokenChars.Modulo */) {
  521. if (isTextStart(scnr)) {
  522. buf += ch;
  523. scnr.next();
  524. }
  525. else {
  526. break;
  527. }
  528. }
  529. else if (ch === CHAR_SP || ch === CHAR_LF) {
  530. if (isTextStart(scnr)) {
  531. buf += ch;
  532. scnr.next();
  533. }
  534. else if (isPluralStart(scnr)) {
  535. break;
  536. }
  537. else {
  538. buf += ch;
  539. scnr.next();
  540. }
  541. }
  542. else {
  543. buf += ch;
  544. scnr.next();
  545. }
  546. }
  547. return buf;
  548. }
  549. function readNamedIdentifier(scnr) {
  550. skipSpaces(scnr);
  551. let ch = '';
  552. let name = '';
  553. while ((ch = takeNamedIdentifierChar(scnr))) {
  554. name += ch;
  555. }
  556. if (scnr.currentChar() === EOF) {
  557. emitError(CompileErrorCodes.UNTERMINATED_CLOSING_BRACE, currentPosition(), 0);
  558. }
  559. return name;
  560. }
  561. function readListIdentifier(scnr) {
  562. skipSpaces(scnr);
  563. let value = '';
  564. if (scnr.currentChar() === '-') {
  565. scnr.next();
  566. value += `-${getDigits(scnr)}`;
  567. }
  568. else {
  569. value += getDigits(scnr);
  570. }
  571. if (scnr.currentChar() === EOF) {
  572. emitError(CompileErrorCodes.UNTERMINATED_CLOSING_BRACE, currentPosition(), 0);
  573. }
  574. return value;
  575. }
  576. function isLiteral(ch) {
  577. return ch !== LITERAL_DELIMITER && ch !== CHAR_LF;
  578. }
  579. function readLiteral(scnr) {
  580. skipSpaces(scnr);
  581. // eslint-disable-next-line no-useless-escape
  582. eat(scnr, `\'`);
  583. let ch = '';
  584. let literal = '';
  585. while ((ch = takeChar(scnr, isLiteral))) {
  586. if (ch === '\\') {
  587. literal += readEscapeSequence(scnr);
  588. }
  589. else {
  590. literal += ch;
  591. }
  592. }
  593. const current = scnr.currentChar();
  594. if (current === CHAR_LF || current === EOF) {
  595. emitError(CompileErrorCodes.UNTERMINATED_SINGLE_QUOTE_IN_PLACEHOLDER, currentPosition(), 0);
  596. // TODO: Is it correct really?
  597. if (current === CHAR_LF) {
  598. scnr.next();
  599. // eslint-disable-next-line no-useless-escape
  600. eat(scnr, `\'`);
  601. }
  602. return literal;
  603. }
  604. // eslint-disable-next-line no-useless-escape
  605. eat(scnr, `\'`);
  606. return literal;
  607. }
  608. function readEscapeSequence(scnr) {
  609. const ch = scnr.currentChar();
  610. switch (ch) {
  611. case '\\':
  612. case `\'`: // eslint-disable-line no-useless-escape
  613. scnr.next();
  614. return `\\${ch}`;
  615. case 'u':
  616. return readUnicodeEscapeSequence(scnr, ch, 4);
  617. case 'U':
  618. return readUnicodeEscapeSequence(scnr, ch, 6);
  619. default:
  620. emitError(CompileErrorCodes.UNKNOWN_ESCAPE_SEQUENCE, currentPosition(), 0, ch);
  621. return '';
  622. }
  623. }
  624. function readUnicodeEscapeSequence(scnr, unicode, digits) {
  625. eat(scnr, unicode);
  626. let sequence = '';
  627. for (let i = 0; i < digits; i++) {
  628. const ch = takeHexDigit(scnr);
  629. if (!ch) {
  630. emitError(CompileErrorCodes.INVALID_UNICODE_ESCAPE_SEQUENCE, currentPosition(), 0, `\\${unicode}${sequence}${scnr.currentChar()}`);
  631. break;
  632. }
  633. sequence += ch;
  634. }
  635. return `\\${unicode}${sequence}`;
  636. }
  637. function isInvalidIdentifier(ch) {
  638. return (ch !== "{" /* TokenChars.BraceLeft */ &&
  639. ch !== "}" /* TokenChars.BraceRight */ &&
  640. ch !== CHAR_SP &&
  641. ch !== CHAR_LF);
  642. }
  643. function readInvalidIdentifier(scnr) {
  644. skipSpaces(scnr);
  645. let ch = '';
  646. let identifiers = '';
  647. while ((ch = takeChar(scnr, isInvalidIdentifier))) {
  648. identifiers += ch;
  649. }
  650. return identifiers;
  651. }
  652. function readLinkedModifier(scnr) {
  653. let ch = '';
  654. let name = '';
  655. while ((ch = takeIdentifierChar(scnr))) {
  656. name += ch;
  657. }
  658. return name;
  659. }
  660. function readLinkedRefer(scnr) {
  661. const fn = (buf) => {
  662. const ch = scnr.currentChar();
  663. if (ch === "{" /* TokenChars.BraceLeft */ ||
  664. ch === "%" /* TokenChars.Modulo */ ||
  665. ch === "@" /* TokenChars.LinkedAlias */ ||
  666. ch === "|" /* TokenChars.Pipe */ ||
  667. ch === "(" /* TokenChars.ParenLeft */ ||
  668. ch === ")" /* TokenChars.ParenRight */ ||
  669. !ch) {
  670. return buf;
  671. }
  672. else if (ch === CHAR_SP) {
  673. return buf;
  674. }
  675. else if (ch === CHAR_LF || ch === DOT) {
  676. buf += ch;
  677. scnr.next();
  678. return fn(buf);
  679. }
  680. else {
  681. buf += ch;
  682. scnr.next();
  683. return fn(buf);
  684. }
  685. };
  686. return fn('');
  687. }
  688. function readPlural(scnr) {
  689. skipSpaces(scnr);
  690. const plural = eat(scnr, "|" /* TokenChars.Pipe */);
  691. skipSpaces(scnr);
  692. return plural;
  693. }
  694. // TODO: We need refactoring of token parsing ...
  695. function readTokenInPlaceholder(scnr, context) {
  696. let token = null;
  697. const ch = scnr.currentChar();
  698. switch (ch) {
  699. case "{" /* TokenChars.BraceLeft */:
  700. if (context.braceNest >= 1) {
  701. emitError(CompileErrorCodes.NOT_ALLOW_NEST_PLACEHOLDER, currentPosition(), 0);
  702. }
  703. scnr.next();
  704. token = getToken(context, 2 /* TokenTypes.BraceLeft */, "{" /* TokenChars.BraceLeft */);
  705. skipSpaces(scnr);
  706. context.braceNest++;
  707. return token;
  708. case "}" /* TokenChars.BraceRight */:
  709. if (context.braceNest > 0 &&
  710. context.currentType === 2 /* TokenTypes.BraceLeft */) {
  711. emitError(CompileErrorCodes.EMPTY_PLACEHOLDER, currentPosition(), 0);
  712. }
  713. scnr.next();
  714. token = getToken(context, 3 /* TokenTypes.BraceRight */, "}" /* TokenChars.BraceRight */);
  715. context.braceNest--;
  716. context.braceNest > 0 && skipSpaces(scnr);
  717. if (context.inLinked && context.braceNest === 0) {
  718. context.inLinked = false;
  719. }
  720. return token;
  721. case "@" /* TokenChars.LinkedAlias */:
  722. if (context.braceNest > 0) {
  723. emitError(CompileErrorCodes.UNTERMINATED_CLOSING_BRACE, currentPosition(), 0);
  724. }
  725. token = readTokenInLinked(scnr, context) || getEndToken(context);
  726. context.braceNest = 0;
  727. return token;
  728. default: {
  729. let validNamedIdentifier = true;
  730. let validListIdentifier = true;
  731. let validLiteral = true;
  732. if (isPluralStart(scnr)) {
  733. if (context.braceNest > 0) {
  734. emitError(CompileErrorCodes.UNTERMINATED_CLOSING_BRACE, currentPosition(), 0);
  735. }
  736. token = getToken(context, 1 /* TokenTypes.Pipe */, readPlural(scnr));
  737. // reset
  738. context.braceNest = 0;
  739. context.inLinked = false;
  740. return token;
  741. }
  742. if (context.braceNest > 0 &&
  743. (context.currentType === 5 /* TokenTypes.Named */ ||
  744. context.currentType === 6 /* TokenTypes.List */ ||
  745. context.currentType === 7 /* TokenTypes.Literal */)) {
  746. emitError(CompileErrorCodes.UNTERMINATED_CLOSING_BRACE, currentPosition(), 0);
  747. context.braceNest = 0;
  748. return readToken(scnr, context);
  749. }
  750. if ((validNamedIdentifier = isNamedIdentifierStart(scnr, context))) {
  751. token = getToken(context, 5 /* TokenTypes.Named */, readNamedIdentifier(scnr));
  752. skipSpaces(scnr);
  753. return token;
  754. }
  755. if ((validListIdentifier = isListIdentifierStart(scnr, context))) {
  756. token = getToken(context, 6 /* TokenTypes.List */, readListIdentifier(scnr));
  757. skipSpaces(scnr);
  758. return token;
  759. }
  760. if ((validLiteral = isLiteralStart(scnr, context))) {
  761. token = getToken(context, 7 /* TokenTypes.Literal */, readLiteral(scnr));
  762. skipSpaces(scnr);
  763. return token;
  764. }
  765. if (!validNamedIdentifier && !validListIdentifier && !validLiteral) {
  766. // TODO: we should be re-designed invalid cases, when we will extend message syntax near the future ...
  767. token = getToken(context, 13 /* TokenTypes.InvalidPlace */, readInvalidIdentifier(scnr));
  768. emitError(CompileErrorCodes.INVALID_TOKEN_IN_PLACEHOLDER, currentPosition(), 0, token.value);
  769. skipSpaces(scnr);
  770. return token;
  771. }
  772. break;
  773. }
  774. }
  775. return token;
  776. }
  777. // TODO: We need refactoring of token parsing ...
  778. function readTokenInLinked(scnr, context) {
  779. const { currentType } = context;
  780. let token = null;
  781. const ch = scnr.currentChar();
  782. if ((currentType === 8 /* TokenTypes.LinkedAlias */ ||
  783. currentType === 9 /* TokenTypes.LinkedDot */ ||
  784. currentType === 12 /* TokenTypes.LinkedModifier */ ||
  785. currentType === 10 /* TokenTypes.LinkedDelimiter */) &&
  786. (ch === CHAR_LF || ch === CHAR_SP)) {
  787. emitError(CompileErrorCodes.INVALID_LINKED_FORMAT, currentPosition(), 0);
  788. }
  789. switch (ch) {
  790. case "@" /* TokenChars.LinkedAlias */:
  791. scnr.next();
  792. token = getToken(context, 8 /* TokenTypes.LinkedAlias */, "@" /* TokenChars.LinkedAlias */);
  793. context.inLinked = true;
  794. return token;
  795. case "." /* TokenChars.LinkedDot */:
  796. skipSpaces(scnr);
  797. scnr.next();
  798. return getToken(context, 9 /* TokenTypes.LinkedDot */, "." /* TokenChars.LinkedDot */);
  799. case ":" /* TokenChars.LinkedDelimiter */:
  800. skipSpaces(scnr);
  801. scnr.next();
  802. return getToken(context, 10 /* TokenTypes.LinkedDelimiter */, ":" /* TokenChars.LinkedDelimiter */);
  803. default:
  804. if (isPluralStart(scnr)) {
  805. token = getToken(context, 1 /* TokenTypes.Pipe */, readPlural(scnr));
  806. // reset
  807. context.braceNest = 0;
  808. context.inLinked = false;
  809. return token;
  810. }
  811. if (isLinkedDotStart(scnr, context) ||
  812. isLinkedDelimiterStart(scnr, context)) {
  813. skipSpaces(scnr);
  814. return readTokenInLinked(scnr, context);
  815. }
  816. if (isLinkedModifierStart(scnr, context)) {
  817. skipSpaces(scnr);
  818. return getToken(context, 12 /* TokenTypes.LinkedModifier */, readLinkedModifier(scnr));
  819. }
  820. if (isLinkedReferStart(scnr, context)) {
  821. skipSpaces(scnr);
  822. if (ch === "{" /* TokenChars.BraceLeft */) {
  823. // scan the placeholder
  824. return readTokenInPlaceholder(scnr, context) || token;
  825. }
  826. else {
  827. return getToken(context, 11 /* TokenTypes.LinkedKey */, readLinkedRefer(scnr));
  828. }
  829. }
  830. if (currentType === 8 /* TokenTypes.LinkedAlias */) {
  831. emitError(CompileErrorCodes.INVALID_LINKED_FORMAT, currentPosition(), 0);
  832. }
  833. context.braceNest = 0;
  834. context.inLinked = false;
  835. return readToken(scnr, context);
  836. }
  837. }
  838. // TODO: We need refactoring of token parsing ...
  839. function readToken(scnr, context) {
  840. let token = { type: 14 /* TokenTypes.EOF */ };
  841. if (context.braceNest > 0) {
  842. return readTokenInPlaceholder(scnr, context) || getEndToken(context);
  843. }
  844. if (context.inLinked) {
  845. return readTokenInLinked(scnr, context) || getEndToken(context);
  846. }
  847. const ch = scnr.currentChar();
  848. switch (ch) {
  849. case "{" /* TokenChars.BraceLeft */:
  850. return readTokenInPlaceholder(scnr, context) || getEndToken(context);
  851. case "}" /* TokenChars.BraceRight */:
  852. emitError(CompileErrorCodes.UNBALANCED_CLOSING_BRACE, currentPosition(), 0);
  853. scnr.next();
  854. return getToken(context, 3 /* TokenTypes.BraceRight */, "}" /* TokenChars.BraceRight */);
  855. case "@" /* TokenChars.LinkedAlias */:
  856. return readTokenInLinked(scnr, context) || getEndToken(context);
  857. default: {
  858. if (isPluralStart(scnr)) {
  859. token = getToken(context, 1 /* TokenTypes.Pipe */, readPlural(scnr));
  860. // reset
  861. context.braceNest = 0;
  862. context.inLinked = false;
  863. return token;
  864. }
  865. const { isModulo, hasSpace } = detectModuloStart(scnr);
  866. if (isModulo) {
  867. return hasSpace
  868. ? getToken(context, 0 /* TokenTypes.Text */, readText(scnr))
  869. : getToken(context, 4 /* TokenTypes.Modulo */, readModulo(scnr));
  870. }
  871. if (isTextStart(scnr)) {
  872. return getToken(context, 0 /* TokenTypes.Text */, readText(scnr));
  873. }
  874. break;
  875. }
  876. }
  877. return token;
  878. }
  879. function nextToken() {
  880. const { currentType, offset, startLoc, endLoc } = _context;
  881. _context.lastType = currentType;
  882. _context.lastOffset = offset;
  883. _context.lastStartLoc = startLoc;
  884. _context.lastEndLoc = endLoc;
  885. _context.offset = currentOffset();
  886. _context.startLoc = currentPosition();
  887. if (_scnr.currentChar() === EOF) {
  888. return getToken(_context, 14 /* TokenTypes.EOF */);
  889. }
  890. return readToken(_scnr, _context);
  891. }
  892. return {
  893. nextToken,
  894. currentOffset,
  895. currentPosition,
  896. context
  897. };
  898. }
  899. const ERROR_DOMAIN$2 = 'parser';
  900. // Backslash backslash, backslash quote, uHHHH, UHHHHHH.
  901. const KNOWN_ESCAPES = /(?:\\\\|\\'|\\u([0-9a-fA-F]{4})|\\U([0-9a-fA-F]{6}))/g;
  902. function fromEscapeSequence(match, codePoint4, codePoint6) {
  903. switch (match) {
  904. case `\\\\`:
  905. return `\\`;
  906. // eslint-disable-next-line no-useless-escape
  907. case `\\\'`:
  908. // eslint-disable-next-line no-useless-escape
  909. return `\'`;
  910. default: {
  911. const codePoint = parseInt(codePoint4 || codePoint6, 16);
  912. if (codePoint <= 0xd7ff || codePoint >= 0xe000) {
  913. return String.fromCodePoint(codePoint);
  914. }
  915. // invalid ...
  916. // Replace them with U+FFFD REPLACEMENT CHARACTER.
  917. return '�';
  918. }
  919. }
  920. }
  921. function createParser(options = {}) {
  922. const location = options.location !== false;
  923. const { onError, onWarn } = options;
  924. function emitError(tokenzer, code, start, offset, ...args) {
  925. const end = tokenzer.currentPosition();
  926. end.offset += offset;
  927. end.column += offset;
  928. if (onError) {
  929. const loc = location ? createLocation(start, end) : null;
  930. const err = createCompileError(code, loc, {
  931. domain: ERROR_DOMAIN$2,
  932. args
  933. });
  934. onError(err);
  935. }
  936. }
  937. function emitWarn(tokenzer, code, start, offset, ...args) {
  938. const end = tokenzer.currentPosition();
  939. end.offset += offset;
  940. end.column += offset;
  941. if (onWarn) {
  942. const loc = location ? createLocation(start, end) : null;
  943. onWarn(createCompileWarn(code, loc, args));
  944. }
  945. }
  946. function startNode(type, offset, loc) {
  947. const node = { type };
  948. if (location) {
  949. node.start = offset;
  950. node.end = offset;
  951. node.loc = { start: loc, end: loc };
  952. }
  953. return node;
  954. }
  955. function endNode(node, offset, pos, type) {
  956. if (type) {
  957. node.type = type;
  958. }
  959. if (location) {
  960. node.end = offset;
  961. if (node.loc) {
  962. node.loc.end = pos;
  963. }
  964. }
  965. }
  966. function parseText(tokenizer, value) {
  967. const context = tokenizer.context();
  968. const node = startNode(3 /* NodeTypes.Text */, context.offset, context.startLoc);
  969. node.value = value;
  970. endNode(node, tokenizer.currentOffset(), tokenizer.currentPosition());
  971. return node;
  972. }
  973. function parseList(tokenizer, index) {
  974. const context = tokenizer.context();
  975. const { lastOffset: offset, lastStartLoc: loc } = context; // get brace left loc
  976. const node = startNode(5 /* NodeTypes.List */, offset, loc);
  977. node.index = parseInt(index, 10);
  978. tokenizer.nextToken(); // skip brach right
  979. endNode(node, tokenizer.currentOffset(), tokenizer.currentPosition());
  980. return node;
  981. }
  982. function parseNamed(tokenizer, key, modulo) {
  983. const context = tokenizer.context();
  984. const { lastOffset: offset, lastStartLoc: loc } = context; // get brace left loc
  985. const node = startNode(4 /* NodeTypes.Named */, offset, loc);
  986. node.key = key;
  987. if (modulo === true) {
  988. node.modulo = true;
  989. }
  990. tokenizer.nextToken(); // skip brach right
  991. endNode(node, tokenizer.currentOffset(), tokenizer.currentPosition());
  992. return node;
  993. }
  994. function parseLiteral(tokenizer, value) {
  995. const context = tokenizer.context();
  996. const { lastOffset: offset, lastStartLoc: loc } = context; // get brace left loc
  997. const node = startNode(9 /* NodeTypes.Literal */, offset, loc);
  998. node.value = value.replace(KNOWN_ESCAPES, fromEscapeSequence);
  999. tokenizer.nextToken(); // skip brach right
  1000. endNode(node, tokenizer.currentOffset(), tokenizer.currentPosition());
  1001. return node;
  1002. }
  1003. function parseLinkedModifier(tokenizer) {
  1004. const token = tokenizer.nextToken();
  1005. const context = tokenizer.context();
  1006. const { lastOffset: offset, lastStartLoc: loc } = context; // get linked dot loc
  1007. const node = startNode(8 /* NodeTypes.LinkedModifier */, offset, loc);
  1008. if (token.type !== 12 /* TokenTypes.LinkedModifier */) {
  1009. // empty modifier
  1010. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_EMPTY_LINKED_MODIFIER, context.lastStartLoc, 0);
  1011. node.value = '';
  1012. endNode(node, offset, loc);
  1013. return {
  1014. nextConsumeToken: token,
  1015. node
  1016. };
  1017. }
  1018. // check token
  1019. if (token.value == null) {
  1020. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1021. }
  1022. node.value = token.value || '';
  1023. endNode(node, tokenizer.currentOffset(), tokenizer.currentPosition());
  1024. return {
  1025. node
  1026. };
  1027. }
  1028. function parseLinkedKey(tokenizer, value) {
  1029. const context = tokenizer.context();
  1030. const node = startNode(7 /* NodeTypes.LinkedKey */, context.offset, context.startLoc);
  1031. node.value = value;
  1032. endNode(node, tokenizer.currentOffset(), tokenizer.currentPosition());
  1033. return node;
  1034. }
  1035. function parseLinked(tokenizer) {
  1036. const context = tokenizer.context();
  1037. const linkedNode = startNode(6 /* NodeTypes.Linked */, context.offset, context.startLoc);
  1038. let token = tokenizer.nextToken();
  1039. if (token.type === 9 /* TokenTypes.LinkedDot */) {
  1040. const parsed = parseLinkedModifier(tokenizer);
  1041. linkedNode.modifier = parsed.node;
  1042. token = parsed.nextConsumeToken || tokenizer.nextToken();
  1043. }
  1044. // asset check token
  1045. if (token.type !== 10 /* TokenTypes.LinkedDelimiter */) {
  1046. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1047. }
  1048. token = tokenizer.nextToken();
  1049. // skip brace left
  1050. if (token.type === 2 /* TokenTypes.BraceLeft */) {
  1051. token = tokenizer.nextToken();
  1052. }
  1053. switch (token.type) {
  1054. case 11 /* TokenTypes.LinkedKey */:
  1055. if (token.value == null) {
  1056. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1057. }
  1058. linkedNode.key = parseLinkedKey(tokenizer, token.value || '');
  1059. break;
  1060. case 5 /* TokenTypes.Named */:
  1061. if (token.value == null) {
  1062. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1063. }
  1064. linkedNode.key = parseNamed(tokenizer, token.value || '');
  1065. break;
  1066. case 6 /* TokenTypes.List */:
  1067. if (token.value == null) {
  1068. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1069. }
  1070. linkedNode.key = parseList(tokenizer, token.value || '');
  1071. break;
  1072. case 7 /* TokenTypes.Literal */:
  1073. if (token.value == null) {
  1074. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1075. }
  1076. linkedNode.key = parseLiteral(tokenizer, token.value || '');
  1077. break;
  1078. default: {
  1079. // empty key
  1080. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_EMPTY_LINKED_KEY, context.lastStartLoc, 0);
  1081. const nextContext = tokenizer.context();
  1082. const emptyLinkedKeyNode = startNode(7 /* NodeTypes.LinkedKey */, nextContext.offset, nextContext.startLoc);
  1083. emptyLinkedKeyNode.value = '';
  1084. endNode(emptyLinkedKeyNode, nextContext.offset, nextContext.startLoc);
  1085. linkedNode.key = emptyLinkedKeyNode;
  1086. endNode(linkedNode, nextContext.offset, nextContext.startLoc);
  1087. return {
  1088. nextConsumeToken: token,
  1089. node: linkedNode
  1090. };
  1091. }
  1092. }
  1093. endNode(linkedNode, tokenizer.currentOffset(), tokenizer.currentPosition());
  1094. return {
  1095. node: linkedNode
  1096. };
  1097. }
  1098. function parseMessage(tokenizer) {
  1099. const context = tokenizer.context();
  1100. const startOffset = context.currentType === 1 /* TokenTypes.Pipe */
  1101. ? tokenizer.currentOffset()
  1102. : context.offset;
  1103. const startLoc = context.currentType === 1 /* TokenTypes.Pipe */
  1104. ? context.endLoc
  1105. : context.startLoc;
  1106. const node = startNode(2 /* NodeTypes.Message */, startOffset, startLoc);
  1107. node.items = [];
  1108. let nextToken = null;
  1109. let modulo = null;
  1110. do {
  1111. const token = nextToken || tokenizer.nextToken();
  1112. nextToken = null;
  1113. switch (token.type) {
  1114. case 0 /* TokenTypes.Text */:
  1115. if (token.value == null) {
  1116. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1117. }
  1118. node.items.push(parseText(tokenizer, token.value || ''));
  1119. break;
  1120. case 6 /* TokenTypes.List */:
  1121. if (token.value == null) {
  1122. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1123. }
  1124. node.items.push(parseList(tokenizer, token.value || ''));
  1125. break;
  1126. case 4 /* TokenTypes.Modulo */:
  1127. modulo = true;
  1128. break;
  1129. case 5 /* TokenTypes.Named */:
  1130. if (token.value == null) {
  1131. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1132. }
  1133. node.items.push(parseNamed(tokenizer, token.value || '', !!modulo));
  1134. if (modulo) {
  1135. emitWarn(tokenizer, CompileWarnCodes.USE_MODULO_SYNTAX, context.lastStartLoc, 0, getTokenCaption(token));
  1136. modulo = null;
  1137. }
  1138. break;
  1139. case 7 /* TokenTypes.Literal */:
  1140. if (token.value == null) {
  1141. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, getTokenCaption(token));
  1142. }
  1143. node.items.push(parseLiteral(tokenizer, token.value || ''));
  1144. break;
  1145. case 8 /* TokenTypes.LinkedAlias */: {
  1146. const parsed = parseLinked(tokenizer);
  1147. node.items.push(parsed.node);
  1148. nextToken = parsed.nextConsumeToken || null;
  1149. break;
  1150. }
  1151. }
  1152. } while (context.currentType !== 14 /* TokenTypes.EOF */ &&
  1153. context.currentType !== 1 /* TokenTypes.Pipe */);
  1154. // adjust message node loc
  1155. const endOffset = context.currentType === 1 /* TokenTypes.Pipe */
  1156. ? context.lastOffset
  1157. : tokenizer.currentOffset();
  1158. const endLoc = context.currentType === 1 /* TokenTypes.Pipe */
  1159. ? context.lastEndLoc
  1160. : tokenizer.currentPosition();
  1161. endNode(node, endOffset, endLoc);
  1162. return node;
  1163. }
  1164. function parsePlural(tokenizer, offset, loc, msgNode) {
  1165. const context = tokenizer.context();
  1166. let hasEmptyMessage = msgNode.items.length === 0;
  1167. const node = startNode(1 /* NodeTypes.Plural */, offset, loc);
  1168. node.cases = [];
  1169. node.cases.push(msgNode);
  1170. do {
  1171. const msg = parseMessage(tokenizer);
  1172. if (!hasEmptyMessage) {
  1173. hasEmptyMessage = msg.items.length === 0;
  1174. }
  1175. node.cases.push(msg);
  1176. } while (context.currentType !== 14 /* TokenTypes.EOF */);
  1177. if (hasEmptyMessage) {
  1178. emitError(tokenizer, CompileErrorCodes.MUST_HAVE_MESSAGES_IN_PLURAL, loc, 0);
  1179. }
  1180. endNode(node, tokenizer.currentOffset(), tokenizer.currentPosition());
  1181. return node;
  1182. }
  1183. function parseResource(tokenizer) {
  1184. const context = tokenizer.context();
  1185. const { offset, startLoc } = context;
  1186. const msgNode = parseMessage(tokenizer);
  1187. if (context.currentType === 14 /* TokenTypes.EOF */) {
  1188. return msgNode;
  1189. }
  1190. else {
  1191. return parsePlural(tokenizer, offset, startLoc, msgNode);
  1192. }
  1193. }
  1194. function parse(source) {
  1195. const tokenizer = createTokenizer(source, assign({}, options));
  1196. const context = tokenizer.context();
  1197. const node = startNode(0 /* NodeTypes.Resource */, context.offset, context.startLoc);
  1198. if (location && node.loc) {
  1199. node.loc.source = source;
  1200. }
  1201. node.body = parseResource(tokenizer);
  1202. if (options.onCacheKey) {
  1203. node.cacheKey = options.onCacheKey(source);
  1204. }
  1205. // assert whether achieved to EOF
  1206. if (context.currentType !== 14 /* TokenTypes.EOF */) {
  1207. emitError(tokenizer, CompileErrorCodes.UNEXPECTED_LEXICAL_ANALYSIS, context.lastStartLoc, 0, source[context.offset] || '');
  1208. }
  1209. endNode(node, tokenizer.currentOffset(), tokenizer.currentPosition());
  1210. return node;
  1211. }
  1212. return { parse };
  1213. }
  1214. function getTokenCaption(token) {
  1215. if (token.type === 14 /* TokenTypes.EOF */) {
  1216. return 'EOF';
  1217. }
  1218. const name = (token.value || '').replace(/\r?\n/gu, '\\n');
  1219. return name.length > 10 ? name.slice(0, 9) + '…' : name;
  1220. }
  1221. function createTransformer(ast, options = {} // eslint-disable-line
  1222. ) {
  1223. const _context = {
  1224. ast,
  1225. helpers: new Set()
  1226. };
  1227. const context = () => _context;
  1228. const helper = (name) => {
  1229. _context.helpers.add(name);
  1230. return name;
  1231. };
  1232. return { context, helper };
  1233. }
  1234. function traverseNodes(nodes, transformer) {
  1235. for (let i = 0; i < nodes.length; i++) {
  1236. traverseNode(nodes[i], transformer);
  1237. }
  1238. }
  1239. function traverseNode(node, transformer) {
  1240. // TODO: if we need pre-hook of transform, should be implemented to here
  1241. switch (node.type) {
  1242. case 1 /* NodeTypes.Plural */:
  1243. traverseNodes(node.cases, transformer);
  1244. transformer.helper("plural" /* HelperNameMap.PLURAL */);
  1245. break;
  1246. case 2 /* NodeTypes.Message */:
  1247. traverseNodes(node.items, transformer);
  1248. break;
  1249. case 6 /* NodeTypes.Linked */: {
  1250. const linked = node;
  1251. traverseNode(linked.key, transformer);
  1252. transformer.helper("linked" /* HelperNameMap.LINKED */);
  1253. transformer.helper("type" /* HelperNameMap.TYPE */);
  1254. break;
  1255. }
  1256. case 5 /* NodeTypes.List */:
  1257. transformer.helper("interpolate" /* HelperNameMap.INTERPOLATE */);
  1258. transformer.helper("list" /* HelperNameMap.LIST */);
  1259. break;
  1260. case 4 /* NodeTypes.Named */:
  1261. transformer.helper("interpolate" /* HelperNameMap.INTERPOLATE */);
  1262. transformer.helper("named" /* HelperNameMap.NAMED */);
  1263. break;
  1264. }
  1265. // TODO: if we need post-hook of transform, should be implemented to here
  1266. }
  1267. // transform AST
  1268. function transform(ast, options = {} // eslint-disable-line
  1269. ) {
  1270. const transformer = createTransformer(ast);
  1271. transformer.helper("normalize" /* HelperNameMap.NORMALIZE */);
  1272. // traverse
  1273. ast.body && traverseNode(ast.body, transformer);
  1274. // set meta information
  1275. const context = transformer.context();
  1276. ast.helpers = Array.from(context.helpers);
  1277. }
  1278. function optimize(ast) {
  1279. const body = ast.body;
  1280. if (body.type === 2 /* NodeTypes.Message */) {
  1281. optimizeMessageNode(body);
  1282. }
  1283. else {
  1284. body.cases.forEach(c => optimizeMessageNode(c));
  1285. }
  1286. return ast;
  1287. }
  1288. function optimizeMessageNode(message) {
  1289. if (message.items.length === 1) {
  1290. const item = message.items[0];
  1291. if (item.type === 3 /* NodeTypes.Text */ || item.type === 9 /* NodeTypes.Literal */) {
  1292. message.static = item.value;
  1293. delete item.value; // optimization for size
  1294. }
  1295. }
  1296. else {
  1297. const values = [];
  1298. for (let i = 0; i < message.items.length; i++) {
  1299. const item = message.items[i];
  1300. if (!(item.type === 3 /* NodeTypes.Text */ || item.type === 9 /* NodeTypes.Literal */)) {
  1301. break;
  1302. }
  1303. if (item.value == null) {
  1304. break;
  1305. }
  1306. values.push(item.value);
  1307. }
  1308. if (values.length === message.items.length) {
  1309. message.static = join(values);
  1310. for (let i = 0; i < message.items.length; i++) {
  1311. const item = message.items[i];
  1312. if (item.type === 3 /* NodeTypes.Text */ || item.type === 9 /* NodeTypes.Literal */) {
  1313. delete item.value; // optimization for size
  1314. }
  1315. }
  1316. }
  1317. }
  1318. }
  1319. const ERROR_DOMAIN$1 = 'minifier';
  1320. /* eslint-disable @typescript-eslint/no-explicit-any */
  1321. function minify(node) {
  1322. node.t = node.type;
  1323. switch (node.type) {
  1324. case 0 /* NodeTypes.Resource */: {
  1325. const resource = node;
  1326. minify(resource.body);
  1327. resource.b = resource.body;
  1328. delete resource.body;
  1329. break;
  1330. }
  1331. case 1 /* NodeTypes.Plural */: {
  1332. const plural = node;
  1333. const cases = plural.cases;
  1334. for (let i = 0; i < cases.length; i++) {
  1335. minify(cases[i]);
  1336. }
  1337. plural.c = cases;
  1338. delete plural.cases;
  1339. break;
  1340. }
  1341. case 2 /* NodeTypes.Message */: {
  1342. const message = node;
  1343. const items = message.items;
  1344. for (let i = 0; i < items.length; i++) {
  1345. minify(items[i]);
  1346. }
  1347. message.i = items;
  1348. delete message.items;
  1349. if (message.static) {
  1350. message.s = message.static;
  1351. delete message.static;
  1352. }
  1353. break;
  1354. }
  1355. case 3 /* NodeTypes.Text */:
  1356. case 9 /* NodeTypes.Literal */:
  1357. case 8 /* NodeTypes.LinkedModifier */:
  1358. case 7 /* NodeTypes.LinkedKey */: {
  1359. const valueNode = node;
  1360. if (valueNode.value) {
  1361. valueNode.v = valueNode.value;
  1362. delete valueNode.value;
  1363. }
  1364. break;
  1365. }
  1366. case 6 /* NodeTypes.Linked */: {
  1367. const linked = node;
  1368. minify(linked.key);
  1369. linked.k = linked.key;
  1370. delete linked.key;
  1371. if (linked.modifier) {
  1372. minify(linked.modifier);
  1373. linked.m = linked.modifier;
  1374. delete linked.modifier;
  1375. }
  1376. break;
  1377. }
  1378. case 5 /* NodeTypes.List */: {
  1379. const list = node;
  1380. list.i = list.index;
  1381. delete list.index;
  1382. break;
  1383. }
  1384. case 4 /* NodeTypes.Named */: {
  1385. const named = node;
  1386. named.k = named.key;
  1387. delete named.key;
  1388. break;
  1389. }
  1390. default:
  1391. {
  1392. throw createCompileError(CompileErrorCodes.UNHANDLED_MINIFIER_NODE_TYPE, null, {
  1393. domain: ERROR_DOMAIN$1,
  1394. args: [node.type]
  1395. });
  1396. }
  1397. }
  1398. delete node.type;
  1399. }
  1400. /* eslint-enable @typescript-eslint/no-explicit-any */
  1401. // eslint-disable-next-line @typescript-eslint/triple-slash-reference
  1402. /// <reference types="source-map-js" />
  1403. const ERROR_DOMAIN = 'parser';
  1404. function createCodeGenerator(ast, options) {
  1405. const { sourceMap, filename, breakLineCode, needIndent: _needIndent } = options;
  1406. const location = options.location !== false;
  1407. const _context = {
  1408. filename,
  1409. code: '',
  1410. column: 1,
  1411. line: 1,
  1412. offset: 0,
  1413. map: undefined,
  1414. breakLineCode,
  1415. needIndent: _needIndent,
  1416. indentLevel: 0
  1417. };
  1418. if (location && ast.loc) {
  1419. _context.source = ast.loc.source;
  1420. }
  1421. const context = () => _context;
  1422. function push(code, node) {
  1423. _context.code += code;
  1424. }
  1425. function _newline(n, withBreakLine = true) {
  1426. const _breakLineCode = withBreakLine ? breakLineCode : '';
  1427. push(_needIndent ? _breakLineCode + ` `.repeat(n) : _breakLineCode);
  1428. }
  1429. function indent(withNewLine = true) {
  1430. const level = ++_context.indentLevel;
  1431. withNewLine && _newline(level);
  1432. }
  1433. function deindent(withNewLine = true) {
  1434. const level = --_context.indentLevel;
  1435. withNewLine && _newline(level);
  1436. }
  1437. function newline() {
  1438. _newline(_context.indentLevel);
  1439. }
  1440. const helper = (key) => `_${key}`;
  1441. const needIndent = () => _context.needIndent;
  1442. return {
  1443. context,
  1444. push,
  1445. indent,
  1446. deindent,
  1447. newline,
  1448. helper,
  1449. needIndent
  1450. };
  1451. }
  1452. function generateLinkedNode(generator, node) {
  1453. const { helper } = generator;
  1454. generator.push(`${helper("linked" /* HelperNameMap.LINKED */)}(`);
  1455. generateNode(generator, node.key);
  1456. if (node.modifier) {
  1457. generator.push(`, `);
  1458. generateNode(generator, node.modifier);
  1459. generator.push(`, _type`);
  1460. }
  1461. else {
  1462. generator.push(`, undefined, _type`);
  1463. }
  1464. generator.push(`)`);
  1465. }
  1466. function generateMessageNode(generator, node) {
  1467. const { helper, needIndent } = generator;
  1468. generator.push(`${helper("normalize" /* HelperNameMap.NORMALIZE */)}([`);
  1469. generator.indent(needIndent());
  1470. const length = node.items.length;
  1471. for (let i = 0; i < length; i++) {
  1472. generateNode(generator, node.items[i]);
  1473. if (i === length - 1) {
  1474. break;
  1475. }
  1476. generator.push(', ');
  1477. }
  1478. generator.deindent(needIndent());
  1479. generator.push('])');
  1480. }
  1481. function generatePluralNode(generator, node) {
  1482. const { helper, needIndent } = generator;
  1483. if (node.cases.length > 1) {
  1484. generator.push(`${helper("plural" /* HelperNameMap.PLURAL */)}([`);
  1485. generator.indent(needIndent());
  1486. const length = node.cases.length;
  1487. for (let i = 0; i < length; i++) {
  1488. generateNode(generator, node.cases[i]);
  1489. if (i === length - 1) {
  1490. break;
  1491. }
  1492. generator.push(', ');
  1493. }
  1494. generator.deindent(needIndent());
  1495. generator.push(`])`);
  1496. }
  1497. }
  1498. function generateResource(generator, node) {
  1499. if (node.body) {
  1500. generateNode(generator, node.body);
  1501. }
  1502. else {
  1503. generator.push('null');
  1504. }
  1505. }
  1506. function generateNode(generator, node) {
  1507. const { helper } = generator;
  1508. switch (node.type) {
  1509. case 0 /* NodeTypes.Resource */:
  1510. generateResource(generator, node);
  1511. break;
  1512. case 1 /* NodeTypes.Plural */:
  1513. generatePluralNode(generator, node);
  1514. break;
  1515. case 2 /* NodeTypes.Message */:
  1516. generateMessageNode(generator, node);
  1517. break;
  1518. case 6 /* NodeTypes.Linked */:
  1519. generateLinkedNode(generator, node);
  1520. break;
  1521. case 8 /* NodeTypes.LinkedModifier */:
  1522. generator.push(JSON.stringify(node.value), node);
  1523. break;
  1524. case 7 /* NodeTypes.LinkedKey */:
  1525. generator.push(JSON.stringify(node.value), node);
  1526. break;
  1527. case 5 /* NodeTypes.List */:
  1528. generator.push(`${helper("interpolate" /* HelperNameMap.INTERPOLATE */)}(${helper("list" /* HelperNameMap.LIST */)}(${node.index}))`, node);
  1529. break;
  1530. case 4 /* NodeTypes.Named */:
  1531. generator.push(`${helper("interpolate" /* HelperNameMap.INTERPOLATE */)}(${helper("named" /* HelperNameMap.NAMED */)}(${JSON.stringify(node.key)}))`, node);
  1532. break;
  1533. case 9 /* NodeTypes.Literal */:
  1534. generator.push(JSON.stringify(node.value), node);
  1535. break;
  1536. case 3 /* NodeTypes.Text */:
  1537. generator.push(JSON.stringify(node.value), node);
  1538. break;
  1539. default:
  1540. {
  1541. throw createCompileError(CompileErrorCodes.UNHANDLED_CODEGEN_NODE_TYPE, null, {
  1542. domain: ERROR_DOMAIN,
  1543. args: [node.type]
  1544. });
  1545. }
  1546. }
  1547. }
  1548. // generate code from AST
  1549. const generate = (ast, options = {} // eslint-disable-line
  1550. ) => {
  1551. const mode = isString(options.mode) ? options.mode : 'normal';
  1552. const filename = isString(options.filename)
  1553. ? options.filename
  1554. : 'message.intl';
  1555. const sourceMap = !!options.sourceMap;
  1556. // prettier-ignore
  1557. const breakLineCode = options.breakLineCode != null
  1558. ? options.breakLineCode
  1559. : mode === 'arrow'
  1560. ? ';'
  1561. : '\n';
  1562. const needIndent = options.needIndent ? options.needIndent : mode !== 'arrow';
  1563. const helpers = ast.helpers || [];
  1564. const generator = createCodeGenerator(ast, {
  1565. mode,
  1566. filename,
  1567. sourceMap,
  1568. breakLineCode,
  1569. needIndent
  1570. });
  1571. generator.push(mode === 'normal' ? `function __msg__ (ctx) {` : `(ctx) => {`);
  1572. generator.indent(needIndent);
  1573. if (helpers.length > 0) {
  1574. generator.push(`const { ${join(helpers.map(s => `${s}: _${s}`), ', ')} } = ctx`);
  1575. generator.newline();
  1576. }
  1577. generator.push(`return `);
  1578. generateNode(generator, ast);
  1579. generator.deindent(needIndent);
  1580. generator.push(`}`);
  1581. delete ast.helpers;
  1582. const { code, map } = generator.context();
  1583. return {
  1584. ast,
  1585. code,
  1586. map: map ? map.toJSON() : undefined // eslint-disable-line @typescript-eslint/no-explicit-any
  1587. };
  1588. };
  1589. function baseCompile(source, options = {}) {
  1590. const assignedOptions = assign({}, options);
  1591. const jit = !!assignedOptions.jit;
  1592. const enalbeMinify = !!assignedOptions.minify;
  1593. const enambeOptimize = assignedOptions.optimize == null ? true : assignedOptions.optimize;
  1594. // parse source codes
  1595. const parser = createParser(assignedOptions);
  1596. const ast = parser.parse(source);
  1597. if (!jit) {
  1598. // transform ASTs
  1599. transform(ast, assignedOptions);
  1600. // generate javascript codes
  1601. return generate(ast, assignedOptions);
  1602. }
  1603. else {
  1604. // optimize ASTs
  1605. enambeOptimize && optimize(ast);
  1606. // minimize ASTs
  1607. enalbeMinify && minify(ast);
  1608. // In JIT mode, no ast transform, no code generation.
  1609. return { ast, code: '' };
  1610. }
  1611. }
  1612. export { CompileErrorCodes, CompileWarnCodes, ERROR_DOMAIN$2 as ERROR_DOMAIN, LOCATION_STUB, baseCompile, createCompileError, createCompileWarn, createLocation, createParser, createPosition, defaultOnError, detectHtmlTag, errorMessages, warnMessages };