1# Copyright 2014 The Chromium Authors. All rights reserved. 2# Use of this source code is governed by a BSD-style license that can be 3# found in the LICENSE file. 4 5from __future__ import absolute_import 6from __future__ import division 7from __future__ import print_function 8 9import os 10import unittest 11 12from py_vulcanize import fake_fs 13from py_vulcanize import generate 14from py_vulcanize import html_generation_controller 15from py_vulcanize import html_module 16from py_vulcanize import parse_html_deps 17from py_vulcanize import project as project_module 18from py_vulcanize import resource 19from py_vulcanize import resource_loader as resource_loader 20import six 21 22 23class ResourceWithFakeContents(resource.Resource): 24 25 def __init__(self, toplevel_dir, absolute_path, fake_contents): 26 """A resource with explicitly provided contents. 27 28 If the resource does not exist, then pass fake_contents=None. This will 29 cause accessing the resource contents to raise an exception mimicking the 30 behavior of regular resources.""" 31 super(ResourceWithFakeContents, self).__init__(toplevel_dir, absolute_path) 32 self._fake_contents = fake_contents 33 34 @property 35 def contents(self): 36 if self._fake_contents is None: 37 raise Exception('File not found') 38 return self._fake_contents 39 40 41class FakeLoader(object): 42 43 def __init__(self, source_paths, initial_filenames_and_contents=None): 44 self._source_paths = source_paths 45 self._file_contents = {} 46 if initial_filenames_and_contents: 47 for k, v in six.iteritems(initial_filenames_and_contents): 48 self._file_contents[k] = v 49 50 def FindResourceGivenAbsolutePath(self, absolute_path): 51 candidate_paths = [] 52 for source_path in self._source_paths: 53 if absolute_path.startswith(source_path): 54 candidate_paths.append(source_path) 55 if len(candidate_paths) == 0: 56 return None 57 58 # Sort by length. Longest match wins. 59 candidate_paths.sort(lambda x, y: len(x) - len(y)) 60 longest_candidate = candidate_paths[-1] 61 62 return ResourceWithFakeContents( 63 longest_candidate, absolute_path, 64 self._file_contents.get(absolute_path, None)) 65 66 def FindResourceGivenRelativePath(self, relative_path): 67 absolute_path = None 68 for script_path in self._source_paths: 69 absolute_path = os.path.join(script_path, relative_path) 70 if absolute_path in self._file_contents: 71 return ResourceWithFakeContents(script_path, absolute_path, 72 self._file_contents[absolute_path]) 73 return None 74 75 76class ParseTests(unittest.TestCase): 77 78 def testValidExternalScriptReferenceToRawScript(self): 79 parse_results = parse_html_deps.HTMLModuleParserResults("""<!DOCTYPE html> 80 <script src="../foo.js"> 81 """) 82 83 file_contents = {} 84 file_contents[os.path.normpath('/tmp/a/foo.js')] = """ 85'i am just some raw script'; 86""" 87 88 metadata = html_module.Parse( 89 FakeLoader([os.path.normpath('/tmp')], file_contents), 90 'a.b.start', 91 '/tmp/a/b/', 92 is_component=False, 93 parser_results=parse_results) 94 self.assertEquals([], metadata.dependent_module_names) 95 self.assertEquals( 96 ['a/foo.js'], metadata.dependent_raw_script_relative_paths) 97 98 def testExternalScriptReferenceToModuleOutsideScriptPath(self): 99 parse_results = parse_html_deps.HTMLModuleParserResults("""<!DOCTYPE html> 100 <script src="/foo.js"> 101 """) 102 103 file_contents = {} 104 file_contents[os.path.normpath('/foo.js')] = '' 105 106 def DoIt(): 107 html_module.Parse(FakeLoader([os.path.normpath('/tmp')], file_contents), 108 'a.b.start', 109 '/tmp/a/b/', 110 is_component=False, 111 parser_results=parse_results) 112 self.assertRaises(Exception, DoIt) 113 114 def testExternalScriptReferenceToFileThatDoesntExist(self): 115 parse_results = parse_html_deps.HTMLModuleParserResults("""<!DOCTYPE html> 116 <script src="/foo.js"> 117 """) 118 119 file_contents = {} 120 121 def DoIt(): 122 html_module.Parse(FakeLoader([os.path.normpath('/tmp')], file_contents), 123 'a.b.start', 124 '/tmp/a/b/', 125 is_component=False, 126 parser_results=parse_results) 127 self.assertRaises(Exception, DoIt) 128 129 def testValidImportOfModule(self): 130 parse_results = parse_html_deps.HTMLModuleParserResults("""<!DOCTYPE html> 131 <link rel="import" href="../foo.html"> 132 """) 133 134 file_contents = {} 135 file_contents[os.path.normpath('/tmp/a/foo.html')] = """ 136""" 137 138 metadata = html_module.Parse( 139 FakeLoader([os.path.normpath('/tmp')], file_contents), 140 'a.b.start', 141 '/tmp/a/b/', 142 is_component=False, 143 parser_results=parse_results) 144 self.assertEquals(['a.foo'], metadata.dependent_module_names) 145 146 def testStyleSheetImport(self): 147 parse_results = parse_html_deps.HTMLModuleParserResults("""<!DOCTYPE html> 148 <link rel="stylesheet" href="../foo.css"> 149 """) 150 151 file_contents = {} 152 file_contents[os.path.normpath('/tmp/a/foo.css')] = """ 153""" 154 metadata = html_module.Parse( 155 FakeLoader([os.path.normpath('/tmp')], file_contents), 156 'a.b.start', 157 '/tmp/a/b/', 158 is_component=False, 159 parser_results=parse_results) 160 self.assertEquals([], metadata.dependent_module_names) 161 self.assertEquals(['a.foo'], metadata.style_sheet_names) 162 163 def testUsingAbsoluteHref(self): 164 parse_results = parse_html_deps.HTMLModuleParserResults("""<!DOCTYPE html> 165 <script src="/foo.js"> 166 """) 167 168 file_contents = {} 169 file_contents[os.path.normpath('/src/foo.js')] = '' 170 171 metadata = html_module.Parse( 172 FakeLoader([os.path.normpath("/tmp"), os.path.normpath("/src")], 173 file_contents), 174 "a.b.start", 175 "/tmp/a/b/", 176 is_component=False, 177 parser_results=parse_results) 178 self.assertEquals(['foo.js'], metadata.dependent_raw_script_relative_paths) 179 180 181class HTMLModuleTests(unittest.TestCase): 182 183 def testBasicModuleGeneration(self): 184 file_contents = {} 185 file_contents[os.path.normpath('/tmp/a/b/start.html')] = """ 186<!DOCTYPE html> 187<link rel="import" href="/widget.html"> 188<link rel="stylesheet" href="../common.css"> 189<script src="/raw_script.js"></script> 190<script src="/excluded_script.js"></script> 191<dom-module id="start"> 192 <template> 193 </template> 194 <script> 195 'use strict'; 196 console.log('inline script for start.html got written'); 197 </script> 198</dom-module> 199""" 200 file_contents[os.path.normpath('/py_vulcanize/py_vulcanize.html')] = """<!DOCTYPE html> 201""" 202 file_contents[os.path.normpath('/components/widget.html')] = """ 203<!DOCTYPE html> 204<link rel="import" href="/py_vulcanize.html"> 205<widget name="widget.html"></widget> 206<script> 207'use strict'; 208console.log('inline script for widget.html'); 209</script> 210""" 211 file_contents[os.path.normpath('/tmp/a/common.css')] = """ 212/* /tmp/a/common.css was written */ 213""" 214 file_contents[os.path.normpath('/raw/raw_script.js')] = """ 215console.log('/raw/raw_script.js was written'); 216""" 217 file_contents[os.path.normpath( 218 '/raw/components/polymer/polymer.min.js')] = """ 219""" 220 221 with fake_fs.FakeFS(file_contents): 222 project = project_module.Project( 223 [os.path.normpath('/py_vulcanize/'), 224 os.path.normpath('/tmp/'), 225 os.path.normpath('/components/'), 226 os.path.normpath('/raw/')]) 227 loader = resource_loader.ResourceLoader(project) 228 a_b_start_module = loader.LoadModule( 229 module_name='a.b.start', excluded_scripts=['\/excluded_script.js']) 230 load_sequence = project.CalcLoadSequenceForModules([a_b_start_module]) 231 232 # Check load sequence names. 233 load_sequence_names = [x.name for x in load_sequence] 234 self.assertEquals(['py_vulcanize', 235 'widget', 236 'a.b.start'], load_sequence_names) 237 238 # Check module_deps on a_b_start_module 239 def HasDependentModule(module, name): 240 return [x for x in module.dependent_modules 241 if x.name == name] 242 assert HasDependentModule(a_b_start_module, 'widget') 243 244 # Check JS generation. 245 js = generate.GenerateJS(load_sequence) 246 assert 'inline script for start.html' in js 247 assert 'inline script for widget.html' in js 248 assert '/raw/raw_script.js' in js 249 assert 'excluded_script.js' not in js 250 251 # Check HTML generation. 252 html = generate.GenerateStandaloneHTMLAsString( 253 load_sequence, title='', flattened_js_url='/blah.js') 254 assert '<dom-module id="start">' in html 255 assert 'inline script for widget.html' not in html 256 assert 'common.css' in html 257 258 def testPolymerConversion(self): 259 file_contents = {} 260 file_contents[os.path.normpath('/tmp/a/b/my_component.html')] = """ 261<!DOCTYPE html> 262<dom-module id="my-component"> 263 <template> 264 </template> 265 <script> 266 'use strict'; 267 Polymer ( { 268 is: "my-component" 269 }); 270 </script> 271</dom-module> 272""" 273 with fake_fs.FakeFS(file_contents): 274 project = project_module.Project([ 275 os.path.normpath('/py_vulcanize/'), os.path.normpath('/tmp/')]) 276 loader = resource_loader.ResourceLoader(project) 277 my_component = loader.LoadModule(module_name='a.b.my_component') 278 279 f = six.StringIO() 280 my_component.AppendJSContentsToFile( 281 f, 282 use_include_tags_for_scripts=False, 283 dir_for_include_tag_root=None) 284 js = f.getvalue().rstrip() 285 expected_js = """ 286 'use strict'; 287 Polymer ( { 288 is: "my-component" 289 }); 290""".rstrip() 291 self.assertEquals(expected_js, js) 292 293 def testInlineStylesheetURLs(self): 294 file_contents = {} 295 file_contents[os.path.normpath('/tmp/a/b/my_component.html')] = """ 296<!DOCTYPE html> 297<style> 298.some-rule { 299 background-image: url('../something.jpg'); 300} 301</style> 302""" 303 file_contents[os.path.normpath('/tmp/a/something.jpg')] = 'jpgdata' 304 with fake_fs.FakeFS(file_contents): 305 project = project_module.Project([ 306 os.path.normpath('/py_vulcanize/'), os.path.normpath('/tmp/')]) 307 loader = resource_loader.ResourceLoader(project) 308 my_component = loader.LoadModule(module_name='a.b.my_component') 309 310 computed_deps = [] 311 my_component.AppendDirectlyDependentFilenamesTo(computed_deps) 312 self.assertEquals(set(computed_deps), 313 set([os.path.normpath('/tmp/a/b/my_component.html'), 314 os.path.normpath('/tmp/a/something.jpg')])) 315 316 f = six.StringIO() 317 ctl = html_generation_controller.HTMLGenerationController() 318 my_component.AppendHTMLContentsToFile(f, ctl) 319 html = f.getvalue().rstrip() 320 # FIXME: This is apparently not used. 321 expected_html = """ 322.some-rule { 323 background-image: url(); 324} 325""".rstrip() 326