| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @license | 
					
						
							|  |  |  |  * Copyright Google Inc. All Rights Reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Use of this source code is governed by an MIT-style license that can be | 
					
						
							|  |  |  |  * found in the LICENSE file at https://angular.io/license
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import * as fs from 'fs'; | 
					
						
							|  |  |  | import * as path from 'path'; | 
					
						
							|  |  |  | import * as ts from 'typescript'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-13 16:55:42 -07:00
										 |  |  | import {mainXi18n} from '../src/extract_i18n'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-21 14:22:06 -07:00
										 |  |  | import {isInBazel, makeTempDir, setup} from './test_support'; | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | function getNgRootDir() { | 
					
						
							|  |  |  |   const moduleFilename = module.filename.replace(/\\/g, '/'); | 
					
						
							|  |  |  |   const distIndex = moduleFilename.indexOf('/dist/all'); | 
					
						
							|  |  |  |   return moduleFilename.substr(0, distIndex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const EXPECTED_XMB = `<?xml version="1.0" encoding="UTF-8" ?>
 | 
					
						
							|  |  |  | <!DOCTYPE messagebundle [ | 
					
						
							|  |  |  | <!ELEMENT messagebundle (msg)*> | 
					
						
							|  |  |  | <!ATTLIST messagebundle class CDATA #IMPLIED> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <!ELEMENT msg (#PCDATA|ph|source)*> | 
					
						
							|  |  |  | <!ATTLIST msg id CDATA #IMPLIED> | 
					
						
							|  |  |  | <!ATTLIST msg seq CDATA #IMPLIED> | 
					
						
							|  |  |  | <!ATTLIST msg name CDATA #IMPLIED> | 
					
						
							|  |  |  | <!ATTLIST msg desc CDATA #IMPLIED> | 
					
						
							|  |  |  | <!ATTLIST msg meaning CDATA #IMPLIED> | 
					
						
							|  |  |  | <!ATTLIST msg obsolete (obsolete) #IMPLIED> | 
					
						
							|  |  |  | <!ATTLIST msg xml:space (default|preserve) "default"> | 
					
						
							|  |  |  | <!ATTLIST msg is_hidden CDATA #IMPLIED> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <!ELEMENT source (#PCDATA)> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <!ELEMENT ph (#PCDATA|ex)*> | 
					
						
							|  |  |  | <!ATTLIST ph name CDATA #REQUIRED> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <!ELEMENT ex (#PCDATA)> | 
					
						
							|  |  |  | ]> | 
					
						
							|  |  |  | <messagebundle> | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |   <msg id="8136548302122759730" desc="desc" meaning="meaning"><source>src/basic.html:1</source><source>src/comp2.ts:1</source><source>src/basic.html:1</source>translate me</msg> | 
					
						
							|  |  |  |   <msg id="9038505069473852515"><source>src/basic.html:3,4</source><source>src/comp2.ts:3,4</source><source>src/comp2.ts:2,3</source><source>src/basic.html:3,4</source> | 
					
						
							|  |  |  |     Welcome</msg> | 
					
						
							| 
									
										
										
										
											2018-07-24 16:24:45 -07:00
										 |  |  |   <msg id="5611534349548281834" desc="with ICU"><source>src/icu.html:1,3</source><source>src/icu.html:5</source>{VAR_PLURAL, plural, =1 {book} other {books} }</msg> | 
					
						
							|  |  |  |   <msg id="5811701742971715242" desc="with ICU and other things"><source>src/icu.html:4,6</source> | 
					
						
							|  |  |  |      foo <ph name="ICU"><ex>{ count, plural, =1 {...} other {...}}</ex>{ count, plural, =1 {...} other {...}}</ph> | 
					
						
							|  |  |  |     </msg> | 
					
						
							|  |  |  |   <msg id="7254052530614200029" desc="with placeholders"><source>src/placeholders.html:1</source>Name: <ph name="START_BOLD_TEXT"><ex><b></ex><b></ph><ph name="NAME"><ex>{{ | 
					
						
							|  |  |  |       name // i18n(ph="name")
 | 
					
						
							|  |  |  |     }}</ex>{{ | 
					
						
							|  |  |  |       name // i18n(ph="name")
 | 
					
						
							|  |  |  |     }}</ph><ph name="CLOSE_BOLD_TEXT"><ex></b></ex></b></ph></msg> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  | </messagebundle> | 
					
						
							|  |  |  | `;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const EXPECTED_XLIFF = `<?xml version="1.0" encoding="UTF-8" ?>
 | 
					
						
							|  |  |  | <xliff version="1.2" xmlns="urn:oasis:names:tc:xliff:document:1.2"> | 
					
						
							|  |  |  |   <file source-language="fr" datatype="plaintext" original="ng2.template"> | 
					
						
							|  |  |  |     <body> | 
					
						
							|  |  |  |       <trans-unit id="76e1eccb1b772fa9f294ef9c146ea6d0efa8a2d4" datatype="html"> | 
					
						
							|  |  |  |         <source>translate me</source> | 
					
						
							|  |  |  |         <context-group purpose="location"> | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |           <context context-type="sourcefile">src/basic.html</context> | 
					
						
							|  |  |  |           <context context-type="linenumber">1</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							|  |  |  |         <context-group purpose="location"> | 
					
						
							|  |  |  |           <context context-type="sourcefile">src/comp2.ts</context> | 
					
						
							|  |  |  |           <context context-type="linenumber">1</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							|  |  |  |         <context-group purpose="location"> | 
					
						
							|  |  |  |           <context context-type="sourcefile">src/basic.html</context> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |           <context context-type="linenumber">1</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							|  |  |  |         <note priority="1" from="description">desc</note> | 
					
						
							|  |  |  |         <note priority="1" from="meaning">meaning</note> | 
					
						
							|  |  |  |       </trans-unit> | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |       <trans-unit id="085a5ecc40cc87451d216725b2befd50866de18a" datatype="html"> | 
					
						
							|  |  |  |         <source> | 
					
						
							|  |  |  |     Welcome</source> | 
					
						
							|  |  |  |         <context-group purpose="location"> | 
					
						
							|  |  |  |           <context context-type="sourcefile">src/basic.html</context> | 
					
						
							|  |  |  |           <context context-type="linenumber">3</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							|  |  |  |         <context-group purpose="location"> | 
					
						
							|  |  |  |           <context context-type="sourcefile">src/comp2.ts</context> | 
					
						
							|  |  |  |           <context context-type="linenumber">3</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |         <context-group purpose="location"> | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |           <context context-type="sourcefile">src/comp2.ts</context> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |           <context context-type="linenumber">2</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |         <context-group purpose="location"> | 
					
						
							|  |  |  |           <context context-type="sourcefile">src/basic.html</context> | 
					
						
							|  |  |  |           <context context-type="linenumber">3</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |       </trans-unit> | 
					
						
							| 
									
										
										
										
											2018-07-24 16:24:45 -07:00
										 |  |  |       <trans-unit id="83937c05b1216e7f4c02a85454260e28fd72d1e3" datatype="html"> | 
					
						
							|  |  |  |         <source>{VAR_PLURAL, plural, =1 {book} other {books} }</source> | 
					
						
							|  |  |  |         <context-group purpose="location"> | 
					
						
							|  |  |  |           <context context-type="sourcefile">src/icu.html</context> | 
					
						
							|  |  |  |           <context context-type="linenumber">1</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							|  |  |  |         <note priority="1" from="description">with ICU</note> | 
					
						
							|  |  |  |       </trans-unit> | 
					
						
							|  |  |  |       <trans-unit id="540c5f481129419ef21017f396b6c2d0869ca4d2" datatype="html"> | 
					
						
							|  |  |  |         <source> | 
					
						
							|  |  |  |      foo <x id="ICU" equiv-text="{ count, plural, =1 {...} other {...}}"/> | 
					
						
							|  |  |  |     </source> | 
					
						
							|  |  |  |         <context-group purpose="location"> | 
					
						
							|  |  |  |           <context context-type="sourcefile">src/icu.html</context> | 
					
						
							|  |  |  |           <context context-type="linenumber">4</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							|  |  |  |         <note priority="1" from="description">with ICU and other things</note> | 
					
						
							|  |  |  |       </trans-unit> | 
					
						
							|  |  |  |       <trans-unit id="ca7678090fddd04441d63b1218177af65f23342d" datatype="html"> | 
					
						
							|  |  |  |         <source>{VAR_PLURAL, plural, =1 {book} other {books} }</source> | 
					
						
							|  |  |  |         <context-group purpose="location"> | 
					
						
							|  |  |  |           <context context-type="sourcefile">src/icu.html</context> | 
					
						
							|  |  |  |           <context context-type="linenumber">5</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							|  |  |  |       </trans-unit> | 
					
						
							|  |  |  |       <trans-unit id="9311399c1ca7c75f771d77acb129e50581c6ec1f" datatype="html"> | 
					
						
							|  |  |  |         <source>Name: <x id="START_BOLD_TEXT" ctype="x-b" equiv-text="<b>"/><x id="NAME" equiv-text="{{ | 
					
						
							|  |  |  |       name // i18n(ph="name")
 | 
					
						
							|  |  |  |     }}"/><x id="CLOSE_BOLD_TEXT" ctype="x-b" equiv-text="</b>"/></source> | 
					
						
							|  |  |  |         <context-group purpose="location"> | 
					
						
							|  |  |  |           <context context-type="sourcefile">src/placeholders.html</context> | 
					
						
							|  |  |  |           <context context-type="linenumber">1</context> | 
					
						
							|  |  |  |         </context-group> | 
					
						
							|  |  |  |         <note priority="1" from="description">with placeholders</note> | 
					
						
							|  |  |  |       </trans-unit> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |     </body> | 
					
						
							|  |  |  |   </file> | 
					
						
							|  |  |  | </xliff> | 
					
						
							|  |  |  | `;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const EXPECTED_XLIFF2 = `<?xml version="1.0" encoding="UTF-8" ?>
 | 
					
						
							|  |  |  | <xliff version="2.0" xmlns="urn:oasis:names:tc:xliff:document:2.0" srcLang="en"> | 
					
						
							|  |  |  |   <file original="ng.template" id="ngi18n"> | 
					
						
							|  |  |  |     <unit id="8136548302122759730"> | 
					
						
							|  |  |  |       <notes> | 
					
						
							|  |  |  |         <note category="description">desc</note> | 
					
						
							|  |  |  |         <note category="meaning">meaning</note> | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |         <note category="location">src/basic.html:1</note> | 
					
						
							|  |  |  |         <note category="location">src/comp2.ts:1</note> | 
					
						
							|  |  |  |         <note category="location">src/basic.html:1</note> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |       </notes> | 
					
						
							|  |  |  |       <segment> | 
					
						
							|  |  |  |         <source>translate me</source> | 
					
						
							|  |  |  |       </segment> | 
					
						
							|  |  |  |     </unit> | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |     <unit id="9038505069473852515"> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |       <notes> | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |         <note category="location">src/basic.html:3,4</note> | 
					
						
							|  |  |  |         <note category="location">src/comp2.ts:3,4</note> | 
					
						
							|  |  |  |         <note category="location">src/comp2.ts:2,3</note> | 
					
						
							|  |  |  |         <note category="location">src/basic.html:3,4</note> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |       </notes> | 
					
						
							|  |  |  |       <segment> | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |         <source> | 
					
						
							|  |  |  |     Welcome</source> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |       </segment> | 
					
						
							|  |  |  |     </unit> | 
					
						
							| 
									
										
										
										
											2018-07-24 16:24:45 -07:00
										 |  |  |     <unit id="5611534349548281834"> | 
					
						
							|  |  |  |       <notes> | 
					
						
							|  |  |  |         <note category="description">with ICU</note> | 
					
						
							|  |  |  |         <note category="location">src/icu.html:1,3</note> | 
					
						
							|  |  |  |         <note category="location">src/icu.html:5</note> | 
					
						
							|  |  |  |       </notes> | 
					
						
							|  |  |  |       <segment> | 
					
						
							|  |  |  |         <source>{VAR_PLURAL, plural, =1 {book} other {books} }</source> | 
					
						
							|  |  |  |       </segment> | 
					
						
							|  |  |  |     </unit> | 
					
						
							|  |  |  |     <unit id="5811701742971715242"> | 
					
						
							|  |  |  |       <notes> | 
					
						
							|  |  |  |         <note category="description">with ICU and other things</note> | 
					
						
							|  |  |  |         <note category="location">src/icu.html:4,6</note> | 
					
						
							|  |  |  |       </notes> | 
					
						
							|  |  |  |       <segment> | 
					
						
							|  |  |  |         <source> | 
					
						
							|  |  |  |      foo <ph id="0" equiv="ICU" disp="{ count, plural, =1 {...} other {...}}"/> | 
					
						
							|  |  |  |     </source> | 
					
						
							|  |  |  |       </segment> | 
					
						
							|  |  |  |     </unit> | 
					
						
							|  |  |  |     <unit id="7254052530614200029"> | 
					
						
							|  |  |  |       <notes> | 
					
						
							|  |  |  |         <note category="description">with placeholders</note> | 
					
						
							|  |  |  |         <note category="location">src/placeholders.html:1</note> | 
					
						
							|  |  |  |       </notes> | 
					
						
							|  |  |  |       <segment> | 
					
						
							|  |  |  |         <source>Name: <pc id="0" equivStart="START_BOLD_TEXT" equivEnd="CLOSE_BOLD_TEXT" type="fmt" dispStart="<b>" dispEnd="</b>"><ph id="1" equiv="NAME" disp="{{ | 
					
						
							|  |  |  |       name // i18n(ph="name")
 | 
					
						
							|  |  |  |     }}"/></pc></source> | 
					
						
							|  |  |  |       </segment> | 
					
						
							|  |  |  |     </unit> | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |   </file> | 
					
						
							|  |  |  | </xliff> | 
					
						
							|  |  |  | `;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | describe('extract_i18n command line', () => { | 
					
						
							|  |  |  |   let basePath: string; | 
					
						
							|  |  |  |   let outDir: string; | 
					
						
							|  |  |  |   let write: (fileName: string, content: string) => void; | 
					
						
							|  |  |  |   let errorSpy: jasmine.Spy&((s: string) => void); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-24 16:24:45 -07:00
										 |  |  |   function writeConfig(tsconfig = '{"extends": "./tsconfig-base.json"}') { | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |     write('tsconfig.json', tsconfig); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   beforeEach(() => { | 
					
						
							|  |  |  |     errorSpy = jasmine.createSpy('consoleError').and.callFake(console.error); | 
					
						
							| 
									
										
										
										
											2018-03-21 14:22:06 -07:00
										 |  |  |     if (isInBazel()) { | 
					
						
							|  |  |  |       const support = setup(); | 
					
						
							|  |  |  |       write = (fileName: string, content: string) => { support.write(fileName, content); }; | 
					
						
							|  |  |  |       basePath = support.basePath; | 
					
						
							|  |  |  |       outDir = path.join(basePath, 'built'); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       basePath = makeTempDir(); | 
					
						
							|  |  |  |       write = (fileName: string, content: string) => { | 
					
						
							|  |  |  |         const dir = path.dirname(fileName); | 
					
						
							| 
									
										
										
										
											2018-07-24 16:24:45 -07:00
										 |  |  |         if (dir !== '.') { | 
					
						
							| 
									
										
										
										
											2018-03-21 14:22:06 -07:00
										 |  |  |           const newDir = path.join(basePath, dir); | 
					
						
							|  |  |  |           if (!fs.existsSync(newDir)) fs.mkdirSync(newDir); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         fs.writeFileSync(path.join(basePath, fileName), content, {encoding: 'utf-8'}); | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |       outDir = path.resolve(basePath, 'built'); | 
					
						
							|  |  |  |       const ngRootDir = getNgRootDir(); | 
					
						
							|  |  |  |       const nodeModulesPath = path.resolve(basePath, 'node_modules'); | 
					
						
							|  |  |  |       fs.mkdirSync(nodeModulesPath); | 
					
						
							|  |  |  |       fs.symlinkSync( | 
					
						
							|  |  |  |           path.resolve(ngRootDir, 'dist', 'all', '@angular'), | 
					
						
							|  |  |  |           path.resolve(nodeModulesPath, '@angular')); | 
					
						
							|  |  |  |       fs.symlinkSync( | 
					
						
							|  |  |  |           path.resolve(ngRootDir, 'node_modules', 'rxjs'), path.resolve(nodeModulesPath, 'rxjs')); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |     write('tsconfig-base.json', `{
 | 
					
						
							|  |  |  |       "compilerOptions": { | 
					
						
							|  |  |  |         "experimentalDecorators": true, | 
					
						
							|  |  |  |         "skipLibCheck": true, | 
					
						
							|  |  |  |         "noImplicitAny": true, | 
					
						
							|  |  |  |         "types": [], | 
					
						
							|  |  |  |         "outDir": "built", | 
					
						
							|  |  |  |         "rootDir": ".", | 
					
						
							|  |  |  |         "baseUrl": ".", | 
					
						
							|  |  |  |         "declaration": true, | 
					
						
							|  |  |  |         "target": "es5", | 
					
						
							|  |  |  |         "module": "es2015", | 
					
						
							|  |  |  |         "moduleResolution": "node", | 
					
						
							|  |  |  |         "lib": ["es6", "dom"], | 
					
						
							|  |  |  |         "typeRoots": ["node_modules/@types"] | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }`);
 | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   function writeSources() { | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |     const welcomeMessage = `
 | 
					
						
							|  |  |  |     <!--i18n--> | 
					
						
							|  |  |  |     Welcome<!--/i18n--> | 
					
						
							|  |  |  |     `;
 | 
					
						
							|  |  |  |     write('src/basic.html', `<div title="translate me" i18n-title="meaning|desc"></div>
 | 
					
						
							|  |  |  |          <p id="welcomeMessage">${welcomeMessage}</p>`);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write('src/comp1.ts', `
 | 
					
						
							|  |  |  |     import {Component} from '@angular/core'; | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @Component({ | 
					
						
							|  |  |  |       selector: 'basic', | 
					
						
							|  |  |  |       templateUrl: './basic.html', | 
					
						
							|  |  |  |     }) | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |     export class BasicCmp1 {}`);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write('src/comp2.ts', `
 | 
					
						
							|  |  |  |     import {Component} from '@angular/core'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @Component({ | 
					
						
							|  |  |  |       selector: 'basic2', | 
					
						
							|  |  |  |       template: \`<div title="translate me" i18n-title="meaning|desc"></div>
 | 
					
						
							|  |  |  |       <p id="welcomeMessage">${welcomeMessage}</p>\`,
 | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     export class BasicCmp2 {} | 
					
						
							|  |  |  |     @Component({ | 
					
						
							|  |  |  |       selector: 'basic4', | 
					
						
							|  |  |  |       template: \`<p id="welcomeMessage">${welcomeMessage}</p>\`,
 | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     export class BasicCmp4 {}`);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write('src/comp3.ts', `
 | 
					
						
							|  |  |  |     import {Component} from '@angular/core'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @Component({ | 
					
						
							|  |  |  |       selector: 'basic3', | 
					
						
							|  |  |  |       templateUrl: './basic.html', | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     export class BasicCmp3 {}`);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-24 16:24:45 -07:00
										 |  |  |     write('src/placeholders.html', `<div i18n="with placeholders">Name: <b>{{
 | 
					
						
							|  |  |  |       name // i18n(ph="name")
 | 
					
						
							|  |  |  |     }}</b></div>`);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write('src/placeholder_cmp.ts', `
 | 
					
						
							|  |  |  |     import {Component} from '@angular/core'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @Component({ | 
					
						
							|  |  |  |       selector: 'placeholders', | 
					
						
							|  |  |  |       templateUrl: './placeholders.html', | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     export class PlaceholderCmp { name = 'whatever'; }`);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write('src/icu.html', `<div i18n="with ICU">{
 | 
					
						
							|  |  |  |       count, plural, =1 {book} other {books} | 
					
						
							|  |  |  |     }</div> | 
					
						
							|  |  |  |     <div i18n="with ICU and other things"> | 
					
						
							|  |  |  |      foo { count, plural, =1 {book} other {books} } | 
					
						
							|  |  |  |     </div>`);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     write('src/icu_cmp.ts', `
 | 
					
						
							|  |  |  |     import {Component} from '@angular/core'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @Component({ | 
					
						
							|  |  |  |       selector: 'icu', | 
					
						
							|  |  |  |       templateUrl: './icu.html', | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     export class IcuCmp { count = 3; }`);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |     write('src/module.ts', `
 | 
					
						
							|  |  |  |     import {NgModule} from '@angular/core'; | 
					
						
							| 
									
										
										
										
											2018-07-24 16:24:45 -07:00
										 |  |  |     import {CommonModule} from '@angular/common'; | 
					
						
							| 
									
										
										
										
											2018-07-13 16:14:08 -07:00
										 |  |  |     import {BasicCmp1} from './comp1'; | 
					
						
							|  |  |  |     import {BasicCmp2, BasicCmp4} from './comp2'; | 
					
						
							|  |  |  |     import {BasicCmp3} from './comp3'; | 
					
						
							| 
									
										
										
										
											2018-07-24 16:24:45 -07:00
										 |  |  |     import {PlaceholderCmp} from './placeholder_cmp'; | 
					
						
							|  |  |  |     import {IcuCmp} from './icu_cmp'; | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @NgModule({ | 
					
						
							| 
									
										
										
										
											2018-07-24 16:24:45 -07:00
										 |  |  |       declarations: [ | 
					
						
							|  |  |  |         BasicCmp1, | 
					
						
							|  |  |  |         BasicCmp2, | 
					
						
							|  |  |  |         BasicCmp3, | 
					
						
							|  |  |  |         BasicCmp4, | 
					
						
							|  |  |  |         PlaceholderCmp, | 
					
						
							|  |  |  |         IcuCmp, | 
					
						
							|  |  |  |       ], | 
					
						
							|  |  |  |       imports: [CommonModule], | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |     }) | 
					
						
							|  |  |  |     export class I18nModule {} | 
					
						
							|  |  |  |     `);
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should extract xmb', () => { | 
					
						
							|  |  |  |     writeConfig(); | 
					
						
							|  |  |  |     writeSources(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const exitCode = | 
					
						
							| 
									
										
										
										
											2017-09-13 16:55:42 -07:00
										 |  |  |         mainXi18n(['-p', basePath, '--i18nFormat=xmb', '--outFile=custom_file.xmb'], errorSpy); | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |     expect(errorSpy).not.toHaveBeenCalled(); | 
					
						
							|  |  |  |     expect(exitCode).toBe(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const xmbOutput = path.join(outDir, 'custom_file.xmb'); | 
					
						
							|  |  |  |     expect(fs.existsSync(xmbOutput)).toBeTruthy(); | 
					
						
							|  |  |  |     const xmb = fs.readFileSync(xmbOutput, {encoding: 'utf-8'}); | 
					
						
							|  |  |  |     expect(xmb).toEqual(EXPECTED_XMB); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should extract xlf', () => { | 
					
						
							|  |  |  |     writeConfig(); | 
					
						
							|  |  |  |     writeSources(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-13 16:55:42 -07:00
										 |  |  |     const exitCode = mainXi18n(['-p', basePath, '--i18nFormat=xlf', '--locale=fr'], errorSpy); | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |     expect(errorSpy).not.toHaveBeenCalled(); | 
					
						
							|  |  |  |     expect(exitCode).toBe(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const xlfOutput = path.join(outDir, 'messages.xlf'); | 
					
						
							|  |  |  |     expect(fs.existsSync(xlfOutput)).toBeTruthy(); | 
					
						
							|  |  |  |     const xlf = fs.readFileSync(xlfOutput, {encoding: 'utf-8'}); | 
					
						
							|  |  |  |     expect(xlf).toEqual(EXPECTED_XLIFF); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should extract xlf2', () => { | 
					
						
							|  |  |  |     writeConfig(); | 
					
						
							|  |  |  |     writeSources(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const exitCode = | 
					
						
							| 
									
										
										
										
											2017-09-13 16:55:42 -07:00
										 |  |  |         mainXi18n(['-p', basePath, '--i18nFormat=xlf2', '--outFile=messages.xliff2.xlf'], errorSpy); | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |     expect(errorSpy).not.toHaveBeenCalled(); | 
					
						
							|  |  |  |     expect(exitCode).toBe(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const xlfOutput = path.join(outDir, 'messages.xliff2.xlf'); | 
					
						
							|  |  |  |     expect(fs.existsSync(xlfOutput)).toBeTruthy(); | 
					
						
							|  |  |  |     const xlf = fs.readFileSync(xlfOutput, {encoding: 'utf-8'}); | 
					
						
							|  |  |  |     expect(xlf).toEqual(EXPECTED_XLIFF2); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should not emit js', () => { | 
					
						
							|  |  |  |     writeConfig(); | 
					
						
							|  |  |  |     writeSources(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const exitCode = | 
					
						
							| 
									
										
										
										
											2017-09-13 16:55:42 -07:00
										 |  |  |         mainXi18n(['-p', basePath, '--i18nFormat=xlf2', '--outFile=messages.xliff2.xlf'], errorSpy); | 
					
						
							| 
									
										
										
										
											2017-09-12 15:53:17 -07:00
										 |  |  |     expect(errorSpy).not.toHaveBeenCalled(); | 
					
						
							|  |  |  |     expect(exitCode).toBe(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const moduleOutput = path.join(outDir, 'src', 'module.js'); | 
					
						
							|  |  |  |     expect(fs.existsSync(moduleOutput)).toBeFalsy(); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2017-09-13 16:55:42 -07:00
										 |  |  | }); |