| OLD | NEW |
| (Empty) |
| 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | |
| 2 // for details. All rights reserved. Use of this source code is governed by a | |
| 3 // BSD-style license that can be found in the LICENSE file. | |
| 4 | |
| 5 library polymer.test.linter_test; | |
| 6 | |
| 7 import 'dart:convert'; | |
| 8 | |
| 9 import 'package:polymer/src/build/common.dart'; | |
| 10 import 'package:polymer/src/build/linter.dart'; | |
| 11 import 'package:polymer/src/build/messages.dart'; | |
| 12 import 'package:unittest/unittest.dart'; | |
| 13 | |
| 14 import 'common.dart'; | |
| 15 | |
| 16 void main() { | |
| 17 _testLinter('nothing to report', { | |
| 18 'a|lib/test.html': '<!DOCTYPE html><html></html>', | |
| 19 }, []); | |
| 20 | |
| 21 group('must have proper initialization imports', () { | |
| 22 _testLinter('nothing to report (no polymer use)', { | |
| 23 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 24 '<script type="application/dart" src="foo.dart">' | |
| 25 '</script>' | |
| 26 '<script src="packages/browser/dart.js"></script>' | |
| 27 '</html>', | |
| 28 }, []); | |
| 29 | |
| 30 _testLinter('nothing to report (no polymer use with import)', { | |
| 31 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 32 '<link rel="import" href="packages/polymer/polymer.html">' | |
| 33 '<script type="application/dart" src="foo.dart">' | |
| 34 '</script>' | |
| 35 '<script src="packages/browser/dart.js"></script>' | |
| 36 '</html>', | |
| 37 }, []); | |
| 38 | |
| 39 _testLinter('nothing to report (polymer used)', { | |
| 40 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 41 '<link rel="import" href="packages/polymer/polymer.html">' | |
| 42 '<polymer-element name="x-a"></polymer-element>' | |
| 43 '<script type="application/dart" src="foo.dart">' | |
| 44 '</script>' | |
| 45 '<script src="packages/browser/dart.js"></script>' | |
| 46 '</html>', | |
| 47 }, []); | |
| 48 | |
| 49 _testLinter('nothing to report (polymer imported transitively)', { | |
| 50 'a|lib/lib.html': '<!DOCTYPE html><html>' | |
| 51 '<link rel="import" href="../../packages/polymer/polymer.html">', | |
| 52 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 53 '<link rel="import" href="packages/a/lib.html">' | |
| 54 '<polymer-element name="x-a"></polymer-element>' | |
| 55 '<script type="application/dart" src="foo.dart">' | |
| 56 '</script>' | |
| 57 '<script src="packages/browser/dart.js"></script>' | |
| 58 '</html>', | |
| 59 }, []); | |
| 60 | |
| 61 test('usePolymerHtmlMessage looks right', () { | |
| 62 _check(int i, String url) { | |
| 63 expect(_usePolymerHtmlMessage(i), | |
| 64 contains('<link rel="import" href="$url">')); | |
| 65 } | |
| 66 _check(0, 'packages/polymer/polymer.html'); | |
| 67 _check(1, '../packages/polymer/polymer.html'); | |
| 68 _check(2, '../../packages/polymer/polymer.html'); | |
| 69 _check(3, '../../../packages/polymer/polymer.html'); | |
| 70 }); | |
| 71 | |
| 72 _testLinter('missing polymer.html in web', { | |
| 73 'a|web/test.html': '<!DOCTYPE html><html>\n' | |
| 74 '<polymer-element name="x-a"></polymer-element>' | |
| 75 '<script type="application/dart" src="foo.dart">' | |
| 76 '</script>' | |
| 77 '<script src="packages/browser/dart.js"></script>' | |
| 78 '</html>', | |
| 79 }, [ | |
| 80 'warning: ${_usePolymerHtmlMessage(0)} ' | |
| 81 '(web/test.html 1 0)', | |
| 82 ]); | |
| 83 | |
| 84 _testLinter('missing polymer.html in web/foo', { | |
| 85 'a|web/foo/test.html': '<!DOCTYPE html><html>\n' | |
| 86 '<polymer-element name="x-a"></polymer-element>' | |
| 87 '<script type="application/dart" src="foo.dart">' | |
| 88 '</script>' | |
| 89 '<script src="packages/browser/dart.js"></script>' | |
| 90 '</html>', | |
| 91 }, [ | |
| 92 'warning: ${_usePolymerHtmlMessage(1)} ' | |
| 93 '(web/foo/test.html 1 0)', | |
| 94 ]); | |
| 95 | |
| 96 _testLinter('missing polymer.html doesn\'t warn in lib', { | |
| 97 'a|lib/test.html': '<!DOCTYPE html><html>\n' | |
| 98 '<polymer-element name="x-a"></polymer-element>' | |
| 99 '<script type="application/dart" src="foo.dart">' | |
| 100 '</script>' | |
| 101 '<script src="packages/browser/dart.js"></script>' | |
| 102 '</html>', | |
| 103 }, []); | |
| 104 | |
| 105 _testLinter('missing polymer.html doesn\'t warn in lib/foo/bar', { | |
| 106 'a|lib/foo/bar/test.html': '<!DOCTYPE html><html>\n' | |
| 107 '<polymer-element name="x-a"></polymer-element>' | |
| 108 '<script type="application/dart" src="foo.dart">' | |
| 109 '</script>' | |
| 110 '<script src="packages/browser/dart.js"></script>' | |
| 111 '</html>', | |
| 112 }, []); | |
| 113 | |
| 114 _testLinter('missing Dart code', { | |
| 115 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 116 '<link rel="import" href="packages/polymer/polymer.html">' | |
| 117 '<script src="packages/browser/dart.js"></script>' | |
| 118 '</html>', | |
| 119 }, ['warning: ${MISSING_INIT_POLYMER.snippet}',]); | |
| 120 | |
| 121 _testLinter('nothing to report, experimental with no Dart code', { | |
| 122 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 123 '<link rel="import" ' | |
| 124 'href="packages/polymer/polymer_experimental.html">' | |
| 125 '<script src="packages/browser/dart.js"></script>' | |
| 126 '</html>', | |
| 127 }, []); | |
| 128 | |
| 129 _testLinter('experimental cannot have Dart code in main document', { | |
| 130 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 131 '<link rel="import" ' | |
| 132 'href="packages/polymer/polymer_experimental.html">\n' | |
| 133 '<script type="application/dart" src="foo.dart">' | |
| 134 '</script>' | |
| 135 '<script src="packages/browser/dart.js"></script>' | |
| 136 '</html>', | |
| 137 }, [ | |
| 138 'warning: ${NO_DART_SCRIPT_AND_EXPERIMENTAL.snippet} ' | |
| 139 '(web/test.html 1 0)', | |
| 140 ]); | |
| 141 | |
| 142 _testLinter('missing Dart code and polymer.html', { | |
| 143 'a|web/test.html': '<!DOCTYPE html><html></html>', | |
| 144 }, ['warning: ${MISSING_INIT_POLYMER.snippet}',]); | |
| 145 | |
| 146 _testLinter('dart_support unnecessary', { | |
| 147 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 148 '<script src="packages/web_components/dart_support.js"></script>' | |
| 149 '<link rel="import" href="../../packages/polymer/polymer.html">' | |
| 150 '<polymer-element name="x-a"></polymer-element>' | |
| 151 '<script type="application/dart" src="foo.dart">' | |
| 152 '</script>' | |
| 153 '<script src="packages/browser/dart.js"></script>' | |
| 154 '</html>', | |
| 155 }, [ | |
| 156 'warning: ${DART_SUPPORT_NO_LONGER_REQUIRED.snippet} ' | |
| 157 '(web/test.html 0 21)' | |
| 158 ]); | |
| 159 | |
| 160 _testLinter('webcomponents unnecessary', { | |
| 161 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 162 '$WEB_COMPONENTS_JS_TAG' | |
| 163 '<script type="application/dart" src="foo.dart">' | |
| 164 '</script>' | |
| 165 '</html>', | |
| 166 }, [ | |
| 167 'warning: ${WEB_COMPONENTS_NO_LONGER_REQUIRED.snippet} ' | |
| 168 '(web/test.html 0 21)' | |
| 169 ]); | |
| 170 | |
| 171 _testLinter('platform.js -> webcomponents.js', { | |
| 172 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 173 '$PLATFORM_JS_TAG' | |
| 174 '<script type="application/dart" src="foo.dart">' | |
| 175 '</script>' | |
| 176 '</html>', | |
| 177 }, [ | |
| 178 'warning: ${PLATFORM_JS_RENAMED.snippet} ' | |
| 179 '(web/test.html 0 21)' | |
| 180 ]); | |
| 181 }); | |
| 182 | |
| 183 group('single script tag per document', () { | |
| 184 _testLinter('two top-level tags', { | |
| 185 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 186 '<link rel="import" href="packages/polymer/polymer.html">' | |
| 187 '<script type="application/dart" src="a.dart">' | |
| 188 '</script>\n' | |
| 189 '<script type="application/dart" src="b.dart">' | |
| 190 '</script>' | |
| 191 '<script src="packages/browser/dart.js"></script>', | |
| 192 }, ['warning: ${ONLY_ONE_TAG.snippet} (web/test.html 1 0)',]); | |
| 193 | |
| 194 _testLinter('two top-level tags, non entrypoint', { | |
| 195 'a|lib/test.html': '<!DOCTYPE html><html>' | |
| 196 '<script type="application/dart" src="a.dart">' | |
| 197 '</script>\n' | |
| 198 '<script type="application/dart" src="b.dart">' | |
| 199 '</script>' | |
| 200 '<script src="packages/browser/dart.js"></script>' | |
| 201 }, ['warning: ${ONLY_ONE_TAG.snippet} (lib/test.html 1 0)',]); | |
| 202 | |
| 203 _testLinter('tags inside elements', { | |
| 204 'a|web/test.html': '<!DOCTYPE html><html>' | |
| 205 '<link rel="import" href="packages/polymer/polymer.html">' | |
| 206 '<polymer-element name="x-a">' | |
| 207 '<script type="application/dart" src="a.dart">' | |
| 208 '</script>' | |
| 209 '</polymer-element>\n' | |
| 210 '<script type="application/dart" src="b.dart">' | |
| 211 '</script>' | |
| 212 '<script src="packages/browser/dart.js"></script>', | |
| 213 }, ['warning: ${ONLY_ONE_TAG.snippet} (web/test.html 1 0)',]); | |
| 214 }); | |
| 215 | |
| 216 group('doctype warning', () { | |
| 217 _testLinter('in web', {'a|web/test.html': '<html></html>',}, [ | |
| 218 'warning: (from html) Unexpected start tag (html). ' | |
| 219 'Expected DOCTYPE. (web/test.html 0 0)', | |
| 220 'warning: ${MISSING_INIT_POLYMER.snippet}', | |
| 221 ]); | |
| 222 | |
| 223 _testLinter('in lib', {'a|lib/test.html': '<html></html>',}, []); | |
| 224 }); | |
| 225 | |
| 226 group('duplicate polymer-elements,', () { | |
| 227 _testLinter('same file', { | |
| 228 'a|lib/test.html': '''<html> | |
| 229 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 230 <polymer-element name="x-a"></polymer-element> | |
| 231 <polymer-element name="x-a"></polymer-element> | |
| 232 </html>'''.replaceAll(' ', ''), | |
| 233 }, [ | |
| 234 'warning: ${DUPLICATE_DEFINITION.create( | |
| 235 {'name': 'x-a', 'second': ''}).snippet} (lib/test.html 2 0)', | |
| 236 'warning: ${DUPLICATE_DEFINITION.create( | |
| 237 {'name': 'x-a', 'second': ' (second definition).'}).snippet} ' | |
| 238 '(lib/test.html 3 0)', | |
| 239 ]); | |
| 240 | |
| 241 _testLinter('other file', { | |
| 242 'a|lib/b.html': '''<html> | |
| 243 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 244 <polymer-element name="x-a"></polymer-element> | |
| 245 </html>'''.replaceAll(' ', ''), | |
| 246 'a|lib/test.html': '''<html> | |
| 247 <link rel="import" href="b.html"> | |
| 248 <polymer-element name="x-a"></polymer-element> | |
| 249 </html>'''.replaceAll(' ', ''), | |
| 250 }, [ | |
| 251 'warning: ${DUPLICATE_DEFINITION.create( | |
| 252 {'name': 'x-a', 'second': ''}).snippet} (lib/b.html 2 0)', | |
| 253 'warning: ${DUPLICATE_DEFINITION.create( | |
| 254 {'name': 'x-a', 'second': ' (second definition).'}).snippet} ' | |
| 255 '(lib/test.html 2 0)', | |
| 256 ]); | |
| 257 | |
| 258 _testLinter('non existing file', { | |
| 259 'a|lib/test.html': '''<html> | |
| 260 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 261 <link rel="import" href="b.html"> | |
| 262 <polymer-element name="x-a"></polymer-element> | |
| 263 </html>'''.replaceAll(' ', ''), | |
| 264 }, [ | |
| 265 'warning: ${IMPORT_NOT_FOUND.create( | |
| 266 {'path': 'lib/b.html', 'package': 'a'}).snippet} ' | |
| 267 '(lib/test.html 2 0)' | |
| 268 ]); | |
| 269 | |
| 270 _testLinter('other package', { | |
| 271 'b|lib/b.html': '''<html> | |
| 272 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 273 <polymer-element name="x-a"></polymer-element> | |
| 274 </html>'''.replaceAll(' ', ''), | |
| 275 'a|lib/test.html': '''<html> | |
| 276 <link rel="import" href="../../packages/b/b.html"> | |
| 277 <polymer-element name="x-a"></polymer-element> | |
| 278 </html>'''.replaceAll(' ', ''), | |
| 279 }, [ | |
| 280 'warning: ${DUPLICATE_DEFINITION.create( | |
| 281 {'name': 'x-a', 'second': ''}).snippet} (package:b/b.html 2 0)', | |
| 282 'warning: ${DUPLICATE_DEFINITION.create( | |
| 283 {'name': 'x-a', 'second': ' (second definition).'}).snippet} ' | |
| 284 '(lib/test.html 2 0)', | |
| 285 ]); | |
| 286 }); | |
| 287 | |
| 288 _testLinter('bad link-rel tag (href missing)', { | |
| 289 'a|lib/test.html': '''<html> | |
| 290 <link rel="import"> | |
| 291 <link rel="stylesheet"> | |
| 292 <link rel="foo"> | |
| 293 <link rel="import" href=""> | |
| 294 </html>'''.replaceAll(' ', ''), | |
| 295 }, [ | |
| 296 'warning: ${MISSING_HREF.create({'rel': 'import'}).snippet} ' | |
| 297 '(lib/test.html 1 0)', | |
| 298 'warning: ${MISSING_HREF.create({'rel': 'stylesheet'}).snippet} ' | |
| 299 '(lib/test.html 2 0)', | |
| 300 'warning: ${MISSING_HREF.create({'rel': 'import'}).snippet} ' | |
| 301 '(lib/test.html 4 0)', | |
| 302 ]); | |
| 303 | |
| 304 _testLinter('<element> is not supported', { | |
| 305 'a|lib/test.html': '''<html> | |
| 306 <element name="x-a"></element> | |
| 307 </html>'''.replaceAll(' ', ''), | |
| 308 }, ['warning: ${ELEMENT_DEPRECATED_EONS_AGO.snippet} (lib/test.html 1 0)']); | |
| 309 | |
| 310 _testLinter('do not nest <polymer-element>', { | |
| 311 'a|lib/test.html': '''<html> | |
| 312 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 313 <polymer-element name="x-a"> | |
| 314 <template><div> | |
| 315 <polymer-element name="b"></polymer-element> | |
| 316 </div></template> | |
| 317 </polymer-element> | |
| 318 </html>'''.replaceAll(' ', ''), | |
| 319 }, ['error: ${NESTED_POLYMER_ELEMENT.snippet} (lib/test.html 4 4)']); | |
| 320 | |
| 321 _testLinter('do put import inside <polymer-element>', { | |
| 322 'a|lib/b.html': '<html></html>', | |
| 323 'a|lib/test.html': '''<html> | |
| 324 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 325 <polymer-element name="x-a"> | |
| 326 <link rel="import" href="b.html"> | |
| 327 <template><div> | |
| 328 </div></template> | |
| 329 </polymer-element> | |
| 330 </html>'''.replaceAll(' ', ''), | |
| 331 }, ['error: ${NO_IMPORT_WITHIN_ELEMENT.snippet} (lib/test.html 3 2)']); | |
| 332 | |
| 333 _testLinter('need a name for <polymer-element>', { | |
| 334 'a|lib/test.html': '''<html> | |
| 335 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 336 <polymer-element></polymer-element> | |
| 337 </html>'''.replaceAll(' ', ''), | |
| 338 }, ['error: ${MISSING_TAG_NAME.snippet} (lib/test.html 2 0)']); | |
| 339 | |
| 340 _testLinter('name for <polymer-element> should have dashes', { | |
| 341 'a|lib/test.html': '''<html> | |
| 342 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 343 <polymer-element name="a"></polymer-element> | |
| 344 </html>'''.replaceAll(' ', ''), | |
| 345 }, [ | |
| 346 'error: ${INVALID_TAG_NAME.create({'name': 'a'}).snippet} ' | |
| 347 '(lib/test.html 2 0)' | |
| 348 ]); | |
| 349 | |
| 350 _testLinter('extend is a valid element or existing tag', { | |
| 351 'a|web/test.html': '''<!DOCTYPE html><html> | |
| 352 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 353 <polymer-element name="x-a" extends="li"></polymer-element> | |
| 354 <script type="application/dart"> | |
| 355 export 'package:polymer/init.dart'; | |
| 356 </script> | |
| 357 </html>'''.replaceAll(' ', ''), | |
| 358 }, []); | |
| 359 | |
| 360 _testLinter('extend is a valid element or existing tag', { | |
| 361 'a|web/test.html': '''<!DOCTYPE html><html> | |
| 362 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 363 <polymer-element name="x-a" extends="x-b"></polymer-element> | |
| 364 <script type="application/dart"> | |
| 365 export 'package:polymer/init.dart'; | |
| 366 </script> | |
| 367 </html>'''.replaceAll(' ', ''), | |
| 368 }, [ | |
| 369 'warning: ${CUSTOM_ELEMENT_NOT_FOUND.create({'tag': 'x-b'}).snippet} ' | |
| 370 '(web/test.html 2 0)' | |
| 371 ]); | |
| 372 | |
| 373 group('script type matches code', () { | |
| 374 _testLinter('top-level, .dart url', { | |
| 375 'a|lib/test.html': '''<html> | |
| 376 <script src="foo.dart"></script> | |
| 377 </html>'''.replaceAll(' ', ''), | |
| 378 }, [ | |
| 379 'warning: Wrong script type, expected type="application/dart".' | |
| 380 ' (lib/test.html 1 0)' | |
| 381 ]); | |
| 382 | |
| 383 _testLinter('in polymer-element, .dart url', { | |
| 384 'a|lib/test.html': '''<html> | |
| 385 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 386 <polymer-element name="x-a"> | |
| 387 <script src="foo.dart"></script> | |
| 388 </polymer-element> | |
| 389 </html>'''.replaceAll(' ', ''), | |
| 390 }, ['warning: ${EXPECTED_DART_MIME_TYPE.snippet} (lib/test.html 3 0)']); | |
| 391 | |
| 392 _testLinter('in polymer-element, .js url', { | |
| 393 'a|lib/test.html': '''<html> | |
| 394 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 395 <polymer-element name="x-a"> | |
| 396 <script src="foo.js"></script> | |
| 397 </polymer-element> | |
| 398 </html>'''.replaceAll(' ', ''), | |
| 399 }, []); | |
| 400 | |
| 401 _testLinter('in polymer-element, inlined', { | |
| 402 'a|lib/test.html': '''<html> | |
| 403 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 404 <polymer-element name="x-a"> | |
| 405 <script>foo...</script> | |
| 406 </polymer-element> | |
| 407 </html>'''.replaceAll(' ', ''), | |
| 408 }, []); | |
| 409 | |
| 410 _testLinter('top-level, dart type & .dart url', { | |
| 411 'a|lib/test.html': '''<html> | |
| 412 <script type="application/dart" src="foo.dart"></script> | |
| 413 </html>'''.replaceAll(' ', ''), | |
| 414 }, []); | |
| 415 | |
| 416 _testLinter('top-level, dart type & .js url', { | |
| 417 'a|lib/test.html': '''<html> | |
| 418 <script type="application/dart" src="foo.js"></script> | |
| 419 </html>'''.replaceAll(' ', ''), | |
| 420 }, ['warning: ${EXPECTED_DART_EXTENSION.snippet} (lib/test.html 1 0)']); | |
| 421 }); | |
| 422 | |
| 423 _testLinter('script tags should have at least src url or inline code', { | |
| 424 'a|lib/test.html': '''<html> | |
| 425 <script type="application/dart"></script> | |
| 426 </html>'''.replaceAll(' ', ''), | |
| 427 }, ['warning: ${SCRIPT_TAG_SEEMS_EMPTY.snippet} (lib/test.html 1 0)']); | |
| 428 | |
| 429 _testLinter('script tags should have only src url or inline code', { | |
| 430 'a|lib/test.html': '''<html> | |
| 431 <script type="application/dart" src="foo.dart">more</script> | |
| 432 </html>'''.replaceAll(' ', ''), | |
| 433 }, [ | |
| 434 'warning: ${FOUND_BOTH_SCRIPT_SRC_AND_TEXT.snippet} (lib/test.html 1 0)' | |
| 435 ]); | |
| 436 | |
| 437 group('event handlers', () { | |
| 438 _testLinter('no longer warn about inline onfoo (Javascript)', { | |
| 439 'a|lib/test.html': '''<html><body> | |
| 440 <div onfoo="something"></div> | |
| 441 '''.replaceAll(' ', ''), | |
| 442 }, []); | |
| 443 | |
| 444 _testLinter('no longer warn about on-foo for auto-binding templates', { | |
| 445 'a|lib/test.html': '''<html><body> | |
| 446 <template is="auto-binding-dart"> | |
| 447 <div on-foo="{{something}}"></div> | |
| 448 <template> | |
| 449 <div>foo</div> | |
| 450 </template> | |
| 451 <div on-foo="{{something}}"></div> | |
| 452 </template> | |
| 453 '''.replaceAll(' ', ''), | |
| 454 }, []); | |
| 455 | |
| 456 _testLinter('on-foo is only supported in polymer elements', { | |
| 457 'a|lib/test.html': '''<html><body> | |
| 458 <div on-foo="{{something}}"></div> | |
| 459 '''.replaceAll(' ', ''), | |
| 460 }, [ | |
| 461 'warning: ${EVENT_HANDLERS_ONLY_WITHIN_POLYMER.snippet} ' | |
| 462 '(lib/test.html 1 5)' | |
| 463 ]); | |
| 464 | |
| 465 _testLinter('on-foo uses the {{ binding }} syntax', { | |
| 466 'a|lib/test.html': '''<html><body> | |
| 467 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 468 <polymer-element name="x-a"><div on-foo="bar"></div> | |
| 469 </polymer-element> | |
| 470 '''.replaceAll(' ', ''), | |
| 471 }, [ | |
| 472 'warning: ${INVALID_EVENT_HANDLER_BODY.create( | |
| 473 {'value': 'bar', 'name': 'on-foo'}).snippet} (lib/test.html 2 33)' | |
| 474 ]); | |
| 475 | |
| 476 _testLinter('on-foo is not an expression', { | |
| 477 'a|lib/test.html': '''<html><body> | |
| 478 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 479 <polymer-element name="x-a"><div on-foo="{{bar()}}"></div> | |
| 480 </polymer-element> | |
| 481 '''.replaceAll(' ', ''), | |
| 482 }, [ | |
| 483 'warning: ${INVALID_EVENT_HANDLER_BODY.create( | |
| 484 {'value': '{{bar()}}', 'name': 'on-foo'}).snippet} ' | |
| 485 '(lib/test.html 2 33)' | |
| 486 ]); | |
| 487 | |
| 488 _testLinter('on-foo can\'t be empty', { | |
| 489 'a|lib/test.html': '''<html><body> | |
| 490 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 491 <polymer-element name="x-a"><div on-foo="{{}}"></div> | |
| 492 </polymer-element> | |
| 493 '''.replaceAll(' ', ''), | |
| 494 }, [ | |
| 495 'warning: ${INVALID_EVENT_HANDLER_BODY.create( | |
| 496 {'value': '{{}}', 'name': 'on-foo'}).snippet} (lib/test.html 2 33)' | |
| 497 ]); | |
| 498 | |
| 499 _testLinter('on-foo can\'t be just space', { | |
| 500 'a|lib/test.html': '''<html><body> | |
| 501 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 502 <polymer-element name="x-a"><div on-foo="{{ }}"></div> | |
| 503 </polymer-element> | |
| 504 '''.replaceAll(' ', ''), | |
| 505 }, [ | |
| 506 'warning: ${INVALID_EVENT_HANDLER_BODY.create( | |
| 507 {'value': '{{ }}', 'name': 'on-foo'}).snippet} (lib/test.html 2 33)' | |
| 508 ]); | |
| 509 | |
| 510 _testLinter('on-foo-bar is supported as a custom event name', { | |
| 511 'a|lib/test.html': '''<html><body> | |
| 512 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 513 <polymer-element name="x-a"><div on-foo-bar="{{quux}}"></div> | |
| 514 </polymer-element> | |
| 515 '''.replaceAll(' ', ''), | |
| 516 }, []); | |
| 517 }); | |
| 518 | |
| 519 group('using custom tags', () { | |
| 520 _testLinter('tag exists (x-tag)', { | |
| 521 'a|web/test.html': '''<!DOCTYPE html> | |
| 522 <x-foo></x-foo> | |
| 523 <script type="application/dart"> | |
| 524 export 'package:polymer/init.dart'; | |
| 525 </script> | |
| 526 '''.replaceAll(' ', ''), | |
| 527 }, [ | |
| 528 'warning: ${CUSTOM_ELEMENT_NOT_FOUND.create({'tag': 'x-foo'}).snippet} ' | |
| 529 '(web/test.html 1 0)' | |
| 530 ]); | |
| 531 | |
| 532 _testLinter('tag exists (type extension)', { | |
| 533 'a|web/test.html': '''<!DOCTYPE html> | |
| 534 <div is="x-foo"></div> | |
| 535 <script type="application/dart"> | |
| 536 export 'package:polymer/init.dart'; | |
| 537 </script>'''.replaceAll(' ', ''), | |
| 538 }, [ | |
| 539 'warning: ${CUSTOM_ELEMENT_NOT_FOUND.create({'tag': 'x-foo'}).snippet} ' | |
| 540 '(web/test.html 1 0)', | |
| 541 ]); | |
| 542 | |
| 543 _testLinter('tag exists (internally defined in code)', { | |
| 544 'a|lib/test.html': '<div is="auto-binding-dart"></div>', | |
| 545 }, []); | |
| 546 | |
| 547 _testLinter('used correctly (no base tag)', { | |
| 548 'a|lib/test.html': ''' | |
| 549 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 550 <polymer-element name="x-a"></polymer-element> | |
| 551 <x-a></x-a> | |
| 552 '''.replaceAll(' ', ''), | |
| 553 }, []); | |
| 554 | |
| 555 _testLinter('used incorrectly (no base tag)', { | |
| 556 'a|lib/test.html': ''' | |
| 557 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 558 <polymer-element name="x-a"></polymer-element> | |
| 559 <div is="x-a"></div> | |
| 560 '''.replaceAll(' ', ''), | |
| 561 }, [ | |
| 562 'warning: ${BAD_INSTANTIATION_BOGUS_BASE_TAG.create( | |
| 563 {'tag': 'x-a', 'base': 'div'}).snippet} (lib/test.html 2 0)' | |
| 564 ]); | |
| 565 | |
| 566 _testLinter('used incorrectly, imported def (no base tag)', { | |
| 567 'a|lib/b.html': ''' | |
| 568 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 569 <polymer-element name="x-a"></polymer-element>''', | |
| 570 'a|lib/test.html': ''' | |
| 571 <link rel="import" href="b.html"> | |
| 572 <div is="x-a"></div> | |
| 573 '''.replaceAll(' ', ''), | |
| 574 }, [ | |
| 575 'warning: ${BAD_INSTANTIATION_BOGUS_BASE_TAG.create( | |
| 576 {'tag': 'x-a', 'base': 'div'}).snippet} (lib/test.html 1 0)' | |
| 577 ]); | |
| 578 | |
| 579 _testLinter('used correctly (base tag)', { | |
| 580 'a|lib/b.html': ''' | |
| 581 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 582 <polymer-element name="x-a" extends="div"> | |
| 583 </polymer-element> | |
| 584 '''.replaceAll(' ', ''), | |
| 585 'a|lib/test.html': ''' | |
| 586 <link rel="import" href="b.html"> | |
| 587 <div is="x-a"></div> | |
| 588 '''.replaceAll(' ', ''), | |
| 589 }, []); | |
| 590 | |
| 591 _testLinter('used incorrectly (missing base tag)', { | |
| 592 'a|lib/b.html': ''' | |
| 593 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 594 <polymer-element name="x-a" extends="div"> | |
| 595 </polymer-element> | |
| 596 '''.replaceAll(' ', ''), | |
| 597 'a|lib/test.html': ''' | |
| 598 <link rel="import" href="b.html"> | |
| 599 <x-a></x-a> | |
| 600 '''.replaceAll(' ', ''), | |
| 601 }, [ | |
| 602 'warning: ${BAD_INSTANTIATION_MISSING_BASE_TAG.create( | |
| 603 {'tag': 'x-a', 'base': 'div'}).snippet} (lib/test.html 1 0)' | |
| 604 ]); | |
| 605 | |
| 606 _testLinter('used incorrectly (wrong base tag)', { | |
| 607 'a|lib/b.html': ''' | |
| 608 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 609 <polymer-element name="x-a" extends="div"> | |
| 610 </polymer-element> | |
| 611 '''.replaceAll(' ', ''), | |
| 612 'a|lib/test.html': ''' | |
| 613 <link rel="import" href="b.html"> | |
| 614 <span is="x-a"></span> | |
| 615 '''.replaceAll(' ', ''), | |
| 616 }, [ | |
| 617 'warning: ${BAD_INSTANTIATION_WRONG_BASE_TAG.create( | |
| 618 {'tag': 'x-a', 'base': 'div'}).snippet} (lib/test.html 1 0)' | |
| 619 ]); | |
| 620 | |
| 621 _testLinter('used incorrectly (wrong base tag, transitive)', { | |
| 622 'a|lib/c.html': ''' | |
| 623 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 624 <polymer-element name="x-c" extends="li"> | |
| 625 </polymer-element> | |
| 626 <polymer-element name="x-b" extends="x-c"> | |
| 627 </polymer-element> | |
| 628 '''.replaceAll(' ', ''), | |
| 629 'a|lib/b.html': ''' | |
| 630 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 631 <link rel="import" href="c.html"> | |
| 632 <polymer-element name="x-a" extends="x-b"> | |
| 633 </polymer-element> | |
| 634 '''.replaceAll(' ', ''), | |
| 635 'a|lib/test.html': ''' | |
| 636 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 637 <link rel="import" href="b.html"> | |
| 638 <span is="x-a"></span> | |
| 639 '''.replaceAll(' ', ''), | |
| 640 }, [ | |
| 641 'warning: ${BAD_INSTANTIATION_WRONG_BASE_TAG.create( | |
| 642 {'tag': 'x-a', 'base': 'li'}).snippet} (lib/test.html 2 0)' | |
| 643 ]); | |
| 644 | |
| 645 _testLinter('FOUC warning works', { | |
| 646 'a|web/a.html': ''' | |
| 647 <!DOCTYPE html> | |
| 648 <html><body> | |
| 649 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 650 <polymer-element name="my-element" noscript></polymer-element> | |
| 651 <my-element>hello!</my-element> | |
| 652 <script type="application/dart"> | |
| 653 export "package:polymer/init.dart"; | |
| 654 </script> | |
| 655 </body></html> | |
| 656 ''', | |
| 657 'a|web/b.html': ''' | |
| 658 <!DOCTYPE html> | |
| 659 <html><body> | |
| 660 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 661 <polymer-element name="my-element" noscript></polymer-element> | |
| 662 <div><my-element>hello!</my-element></div> | |
| 663 <script type="application/dart"> | |
| 664 export "package:polymer/init.dart"; | |
| 665 </script> | |
| 666 </body></html> | |
| 667 ''', | |
| 668 'a|web/c.html': ''' | |
| 669 <!DOCTYPE html> | |
| 670 <html unresolved><body> | |
| 671 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 672 <polymer-element name="my-element" noscript></polymer-element> | |
| 673 <my-element>hello!</my-element> | |
| 674 <script type="application/dart"> | |
| 675 export "package:polymer/init.dart"; | |
| 676 </script> | |
| 677 </body></html> | |
| 678 ''' | |
| 679 }, [ | |
| 680 'warning: ${POSSIBLE_FUOC.snippet} (web/a.html 4 14)', | |
| 681 'warning: ${POSSIBLE_FUOC.snippet} (web/b.html 4 19)', | |
| 682 'warning: ${POSSIBLE_FUOC.snippet} (web/c.html 4 14)', | |
| 683 ]); | |
| 684 | |
| 685 _testLinter('FOUC, no false positives.', { | |
| 686 // Parent has unresolved attribute. | |
| 687 'a|web/a.html': ''' | |
| 688 <!DOCTYPE html> | |
| 689 <html><body> | |
| 690 <div unresolved> | |
| 691 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 692 <polymer-element name="my-element" noscript></polymer-element> | |
| 693 <my-element>hello!</my-element> | |
| 694 </div> | |
| 695 <script type="application/dart"> | |
| 696 export "package:polymer/init.dart"; | |
| 697 </script> | |
| 698 </body></html> | |
| 699 ''', | |
| 700 // Body has unresolved attribute. | |
| 701 'a|web/b.html': ''' | |
| 702 <!DOCTYPE html> | |
| 703 <html><body unresolved> | |
| 704 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 705 <polymer-element name="my-element" noscript></polymer-element> | |
| 706 <my-element>hello!</my-element> | |
| 707 <script type="application/dart"> | |
| 708 export "package:polymer/init.dart"; | |
| 709 </script> | |
| 710 </body></html> | |
| 711 ''', | |
| 712 // Inside polymer-element tags its fine. | |
| 713 'a|web/c.html': ''' | |
| 714 <!DOCTYPE html> | |
| 715 <html><body> | |
| 716 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 717 <polymer-element name="my-element" noscript></polymer-element> | |
| 718 <polymer-element name="foo-element"> | |
| 719 <template><my-element>hello!</my-element></template> | |
| 720 </polymer-element> | |
| 721 <script type="application/dart"> | |
| 722 export "package:polymer/init.dart"; | |
| 723 </script> | |
| 724 </body></html> | |
| 725 ''', | |
| 726 // Empty custom elements are fine. | |
| 727 'a|web/d.html': ''' | |
| 728 <!DOCTYPE html> | |
| 729 <html><body> | |
| 730 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 731 <polymer-element name="my-element" noscript></polymer-element> | |
| 732 <my-element></my-element> | |
| 733 <script type="application/dart"> | |
| 734 export "package:polymer/init.dart"; | |
| 735 </script> | |
| 736 </body></html> | |
| 737 ''', | |
| 738 // Entry points only! | |
| 739 'a|lib/a.html': ''' | |
| 740 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 741 <polymer-element name="my-element" noscript></polymer-element> | |
| 742 <my-element>hello!</my-element> | |
| 743 ''', | |
| 744 }, []); | |
| 745 }); | |
| 746 | |
| 747 group('custom attributes', () { | |
| 748 _testLinter('foo-bar is no longer supported in attributes', { | |
| 749 'a|lib/test.html': '''<html><body> | |
| 750 <link rel="import" href="../../packages/polymer/polymer.html"> | |
| 751 <polymer-element name="x-a" attributes="foo-bar"> | |
| 752 </polymer-element> | |
| 753 '''.replaceAll(' ', ''), | |
| 754 }, [ | |
| 755 'warning: ${NO_DASHES_IN_CUSTOM_ATTRIBUTES.create( | |
| 756 {'name': 'foo-bar', 'alternative': '"fooBar" or "foobar"'}) | |
| 757 .snippet} (lib/test.html 2 28)' | |
| 758 ]); | |
| 759 }); | |
| 760 | |
| 761 _testLinter("namespaced attributes don't cause an internal error", { | |
| 762 'a|lib/test.html': '''<html><body> | |
| 763 <svg xmlns="http://www.w3.org/2000/svg" width="520" height="350"> | |
| 764 </svg> | |
| 765 '''.replaceAll(' ', ''), | |
| 766 }, []); | |
| 767 | |
| 768 group('output logs to file', () { | |
| 769 final outputLogsPhases = [ | |
| 770 [ | |
| 771 new Linter(new TransformOptions( | |
| 772 injectBuildLogsInOutput: true, releaseMode: false)) | |
| 773 ] | |
| 774 ]; | |
| 775 | |
| 776 testPhases("logs are output to file", outputLogsPhases, { | |
| 777 'a|web/test.html': '<!DOCTYPE html><html>\n' | |
| 778 '<polymer-element name="x-a"></polymer-element>' | |
| 779 '<script type="application/dart" src="foo.dart">' | |
| 780 '</script>' | |
| 781 '<script src="packages/browser/dart.js"></script>' | |
| 782 '</html>', | |
| 783 }, { | |
| 784 'a|web/test.html._buildLogs.1': '{"polymer#3":[{' | |
| 785 '"level":"Warning",' | |
| 786 '"message":{' | |
| 787 '"id":"polymer#3",' | |
| 788 '"snippet":"${_usePolymerHtmlMessage(0).replaceAll('"','\\"')}"' | |
| 789 '},' | |
| 790 '"span":{' | |
| 791 '"start":{' | |
| 792 '"url":"web/test.html",' | |
| 793 '"offset":22,' | |
| 794 '"line":1,' | |
| 795 '"column":0' | |
| 796 '},' | |
| 797 '"end":{' | |
| 798 '"url":"web/test.html",' | |
| 799 '"offset":50,' | |
| 800 '"line":1,' | |
| 801 '"column":28' | |
| 802 '},' | |
| 803 '"text":"<polymer-element name=\\"x-a\\">"' | |
| 804 '}' | |
| 805 '}]}', | |
| 806 }, [ | |
| 807 // Logs should still make it to barback too. | |
| 808 'warning: ${_usePolymerHtmlMessage(0)} (web/test.html 1 0)', | |
| 809 ]); | |
| 810 }); | |
| 811 } | |
| 812 | |
| 813 _usePolymerHtmlMessage(int i) { | |
| 814 var prefix = '../' * i; | |
| 815 return USE_POLYMER_HTML.create({'reachOutPrefix': prefix}).snippet; | |
| 816 } | |
| 817 | |
| 818 _testLinter(String name, Map inputFiles, List outputMessages, | |
| 819 [bool solo = false]) { | |
| 820 var outputFiles = {}; | |
| 821 if (outputMessages.every((m) => m.startsWith('warning:'))) { | |
| 822 inputFiles.forEach((k, v) => outputFiles[k] = v); | |
| 823 } | |
| 824 if (outputMessages.isEmpty) { | |
| 825 var linter = new Linter(new TransformOptions()); | |
| 826 testPhases(name, [[linter]], inputFiles, outputFiles, outputMessages, solo); | |
| 827 } else { | |
| 828 testLogOutput((options) => new Linter(options), name, inputFiles, | |
| 829 outputFiles, outputMessages, solo); | |
| 830 } | |
| 831 } | |
| OLD | NEW |