OLD | NEW |
1 /* | 1 /* |
2 * This file is part of Adblock Plus <https://adblockplus.org/>, | 2 * This file is part of Adblock Plus <https://adblockplus.org/>, |
3 * Copyright (C) 2006-present eyeo GmbH | 3 * Copyright (C) 2006-present eyeo GmbH |
4 * | 4 * |
5 * Adblock Plus is free software: you can redistribute it and/or modify | 5 * Adblock Plus is free software: you can redistribute it and/or modify |
6 * it under the terms of the GNU General Public License version 3 as | 6 * it under the terms of the GNU General Public License version 3 as |
7 * published by the Free Software Foundation. | 7 * published by the Free Software Foundation. |
8 * | 8 * |
9 * Adblock Plus is distributed in the hope that it will be useful, | 9 * Adblock Plus is distributed in the hope that it will be useful, |
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of | 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
(...skipping 369 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
380 domainSource: null, | 380 domainSource: null, |
381 | 381 |
382 /** | 382 /** |
383 * Separator character used in domainSource property, must be | 383 * Separator character used in domainSource property, must be |
384 * overridden by subclasses | 384 * overridden by subclasses |
385 * @type {string} | 385 * @type {string} |
386 */ | 386 */ |
387 domainSeparator: null, | 387 domainSeparator: null, |
388 | 388 |
389 /** | 389 /** |
390 * Determines whether domainSource is already upper-case, | 390 * Determines whether domainSource is already lower-case, |
391 * can be overridden by subclasses. | 391 * can be overridden by subclasses. |
392 * @type {boolean} | 392 * @type {boolean} |
393 */ | 393 */ |
394 domainSourceIsUpperCase: false, | 394 domainSourceIsLowerCase: false, |
395 | 395 |
396 /** | 396 /** |
397 * Map containing domains that this filter should match on/not match | 397 * Map containing domains that this filter should match on/not match |
398 * on or null if the filter should match on all domains | 398 * on or null if the filter should match on all domains |
399 * @type {?Map.<string,boolean>} | 399 * @type {?Map.<string,boolean>} |
400 */ | 400 */ |
401 get domains() | 401 get domains() |
402 { | 402 { |
403 let domains = null; | 403 let domains = null; |
404 | 404 |
405 if (this.domainSource) | 405 if (this.domainSource) |
406 { | 406 { |
407 let source = this.domainSource; | 407 let source = this.domainSource; |
408 if (!this.domainSourceIsUpperCase) | 408 if (!this.domainSourceIsLowerCase) |
409 { | 409 { |
410 // RegExpFilter already have uppercase domains | 410 // RegExpFilter already have lowercase domains |
411 source = source.toUpperCase(); | 411 source = source.toLowerCase(); |
412 } | 412 } |
413 let list = source.split(this.domainSeparator); | 413 let list = source.split(this.domainSeparator); |
414 if (list.length == 1 && list[0][0] != "~") | 414 if (list.length == 1 && list[0][0] != "~") |
415 { | 415 { |
416 // Fast track for the common one-domain scenario | 416 // Fast track for the common one-domain scenario |
417 domains = new Map([["", false], [list[0], true]]); | 417 domains = new Map([["", false], [list[0], true]]); |
418 } | 418 } |
419 else | 419 else |
420 { | 420 { |
421 let hasIncludes = false; | 421 let hasIncludes = false; |
(...skipping 56 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
478 | 478 |
479 // If no domains are set the rule matches everywhere | 479 // If no domains are set the rule matches everywhere |
480 if (!this.domains) | 480 if (!this.domains) |
481 return true; | 481 return true; |
482 | 482 |
483 // If the document has no host name, match only if the filter | 483 // If the document has no host name, match only if the filter |
484 // isn't restricted to specific domains | 484 // isn't restricted to specific domains |
485 if (!docDomain) | 485 if (!docDomain) |
486 return this.domains.get(""); | 486 return this.domains.get(""); |
487 | 487 |
488 docDomain = docDomain.replace(/\.+$/, "").toUpperCase(); | 488 docDomain = docDomain.replace(/\.+$/, "").toLowerCase(); |
489 | 489 |
490 while (true) | 490 while (true) |
491 { | 491 { |
492 let isDomainIncluded = this.domains.get(docDomain); | 492 let isDomainIncluded = this.domains.get(docDomain); |
493 if (typeof isDomainIncluded != "undefined") | 493 if (typeof isDomainIncluded != "undefined") |
494 return isDomainIncluded; | 494 return isDomainIncluded; |
495 | 495 |
496 let nextDot = docDomain.indexOf("."); | 496 let nextDot = docDomain.indexOf("."); |
497 if (nextDot < 0) | 497 if (nextDot < 0) |
498 break; | 498 break; |
499 docDomain = docDomain.substr(nextDot + 1); | 499 docDomain = docDomain.substr(nextDot + 1); |
500 } | 500 } |
501 return this.domains.get(""); | 501 return this.domains.get(""); |
502 }, | 502 }, |
503 | 503 |
504 /** | 504 /** |
505 * Checks whether this filter is active only on a domain and its subdomains. | 505 * Checks whether this filter is active only on a domain and its subdomains. |
506 * @param {string} docDomain | 506 * @param {string} docDomain |
507 * @return {boolean} | 507 * @return {boolean} |
508 */ | 508 */ |
509 isActiveOnlyOnDomain(docDomain) | 509 isActiveOnlyOnDomain(docDomain) |
510 { | 510 { |
511 if (!docDomain || !this.domains || this.domains.get("")) | 511 if (!docDomain || !this.domains || this.domains.get("")) |
512 return false; | 512 return false; |
513 | 513 |
514 docDomain = docDomain.replace(/\.+$/, "").toUpperCase(); | 514 docDomain = docDomain.replace(/\.+$/, "").toLowerCase(); |
515 | 515 |
516 for (let [domain, isIncluded] of this.domains) | 516 for (let [domain, isIncluded] of this.domains) |
517 { | 517 { |
518 if (isIncluded && domain != docDomain) | 518 if (isIncluded && domain != docDomain) |
519 { | 519 { |
520 if (domain.length <= docDomain.length) | 520 if (domain.length <= docDomain.length) |
521 return false; | 521 return false; |
522 | 522 |
523 if (!domain.endsWith("." + docDomain)) | 523 if (!domain.endsWith("." + docDomain)) |
524 return false; | 524 return false; |
(...skipping 80 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
605 else | 605 else |
606 { | 606 { |
607 // No need to convert this filter to regular expression yet, do it on demand | 607 // No need to convert this filter to regular expression yet, do it on demand |
608 this.regexpSource = regexpSource; | 608 this.regexpSource = regexpSource; |
609 } | 609 } |
610 } | 610 } |
611 exports.RegExpFilter = RegExpFilter; | 611 exports.RegExpFilter = RegExpFilter; |
612 | 612 |
613 RegExpFilter.prototype = extend(ActiveFilter, { | 613 RegExpFilter.prototype = extend(ActiveFilter, { |
614 /** | 614 /** |
615 * @see ActiveFilter.domainSourceIsUpperCase | 615 * @see ActiveFilter.domainSourceIsLowerCase |
616 */ | 616 */ |
617 domainSourceIsUpperCase: true, | 617 domainSourceIsLowerCase: true, |
618 | 618 |
619 /** | 619 /** |
620 * Number of filters contained, will always be 1 (required to | 620 * Number of filters contained, will always be 1 (required to |
621 * optimize Matcher). | 621 * optimize Matcher). |
622 * @type {number} | 622 * @type {number} |
623 */ | 623 */ |
624 length: 1, | 624 length: 1, |
625 | 625 |
626 /** | 626 /** |
627 * @see ActiveFilter.domainSeparator | 627 * @see ActiveFilter.domainSeparator |
(...skipping 120 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
748 text = match.input.substr(0, match.index); | 748 text = match.input.substr(0, match.index); |
749 for (let option of options) | 749 for (let option of options) |
750 { | 750 { |
751 let value = null; | 751 let value = null; |
752 let separatorIndex = option.indexOf("="); | 752 let separatorIndex = option.indexOf("="); |
753 if (separatorIndex >= 0) | 753 if (separatorIndex >= 0) |
754 { | 754 { |
755 value = option.substr(separatorIndex + 1); | 755 value = option.substr(separatorIndex + 1); |
756 option = option.substr(0, separatorIndex); | 756 option = option.substr(0, separatorIndex); |
757 } | 757 } |
758 option = option.replace(/-/, "_").toUpperCase(); | 758 let type = (option[0] == "~" ? option.substr(1) : option) |
759 if (option in RegExpFilter.typeMap) | 759 .replace(/-/, "_").toUpperCase(); |
| 760 option = option.toLowerCase(); |
| 761 if (type in RegExpFilter.typeMap) |
760 { | 762 { |
761 if (contentType == null) | 763 if (option[0] == "~") |
762 contentType = 0; | 764 { |
763 contentType |= RegExpFilter.typeMap[option]; | 765 if (contentType == null) |
| 766 ({contentType} = RegExpFilter.prototype); |
| 767 contentType &= ~RegExpFilter.typeMap[type]; |
| 768 } |
| 769 else |
| 770 { |
| 771 if (contentType == null) |
| 772 contentType = 0; |
| 773 contentType |= RegExpFilter.typeMap[type]; |
764 | 774 |
765 if (option == "CSP" && value) | 775 if (type == "CSP" && value) |
766 csp = value; | 776 csp = value; |
| 777 } |
767 } | 778 } |
768 else if (option[0] == "~" && option.substr(1) in RegExpFilter.typeMap) | 779 else if (option == "match-case") |
769 { | |
770 if (contentType == null) | |
771 ({contentType} = RegExpFilter.prototype); | |
772 contentType &= ~RegExpFilter.typeMap[option.substr(1)]; | |
773 } | |
774 else if (option == "MATCH_CASE") | |
775 matchCase = true; | 780 matchCase = true; |
776 else if (option == "~MATCH_CASE") | 781 else if (option == "~match-case") |
777 matchCase = false; | 782 matchCase = false; |
778 else if (option == "DOMAIN" && value) | 783 else if (option == "domain" && value) |
779 domains = value.toUpperCase(); | 784 domains = value.toLowerCase(); |
780 else if (option == "THIRD_PARTY") | 785 else if (option == "third-party") |
781 thirdParty = true; | 786 thirdParty = true; |
782 else if (option == "~THIRD_PARTY") | 787 else if (option == "~third-party") |
783 thirdParty = false; | 788 thirdParty = false; |
784 else if (option == "COLLAPSE") | 789 else if (option == "collapse") |
785 collapse = true; | 790 collapse = true; |
786 else if (option == "~COLLAPSE") | 791 else if (option == "~collapse") |
787 collapse = false; | 792 collapse = false; |
788 else if (option == "SITEKEY" && value) | 793 else if (option == "sitekey" && value) |
789 sitekeys = value.toUpperCase(); | 794 sitekeys = value.toUpperCase(); |
790 else if (option == "REWRITE" && value) | 795 else if (option == "rewrite" && value) |
791 rewrite = value; | 796 rewrite = value; |
792 else | 797 else |
793 return new InvalidFilter(origText, "filter_unknown_option"); | 798 return new InvalidFilter(origText, "filter_unknown_option"); |
794 } | 799 } |
795 } | 800 } |
796 | 801 |
797 // For security reasons, never match $rewrite filters | 802 // For security reasons, never match $rewrite filters |
798 // against requests that might load any code to be executed. | 803 // against requests that might load any code to be executed. |
799 if (rewrite != null) | 804 if (rewrite != null) |
800 { | 805 { |
(...skipping 281 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1082 */ | 1087 */ |
1083 function ElemHideEmulationFilter(text, domains, selector) | 1088 function ElemHideEmulationFilter(text, domains, selector) |
1084 { | 1089 { |
1085 ElemHideBase.call(this, text, domains, selector); | 1090 ElemHideBase.call(this, text, domains, selector); |
1086 } | 1091 } |
1087 exports.ElemHideEmulationFilter = ElemHideEmulationFilter; | 1092 exports.ElemHideEmulationFilter = ElemHideEmulationFilter; |
1088 | 1093 |
1089 ElemHideEmulationFilter.prototype = extend(ElemHideBase, { | 1094 ElemHideEmulationFilter.prototype = extend(ElemHideBase, { |
1090 type: "elemhideemulation" | 1095 type: "elemhideemulation" |
1091 }); | 1096 }); |
OLD | NEW |