mirror of
https://github.com/Hopiu/linkchecker.git
synced 2026-03-17 06:20:27 +00:00
92 lines
3 KiB
Python
92 lines
3 KiB
Python
|
|
# Copyright (C) 2022 Stefan Fisk
|
|||
|
|
#
|
|||
|
|
# This program is free software; you can redistribute it and/or modify
|
|||
|
|
# it under the terms of the GNU General Public License as published by
|
|||
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|||
|
|
# (at your option) any later version.
|
|||
|
|
#
|
|||
|
|
# This program is distributed in the hope that it will be useful,
|
|||
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|||
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|||
|
|
# GNU General Public License for more details.
|
|||
|
|
#
|
|||
|
|
# You should have received a copy of the GNU General Public License along
|
|||
|
|
# with this program; if not, write to the Free Software Foundation, Inc.,
|
|||
|
|
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|||
|
|
"""
|
|||
|
|
srcset attribute parser
|
|||
|
|
"""
|
|||
|
|
|
|||
|
|
|
|||
|
|
_TAB = '\u0009'
|
|||
|
|
_LF = '\u000A'
|
|||
|
|
_FF = '\u000C'
|
|||
|
|
_CR = '\u000D'
|
|||
|
|
_SPACE = '\u0020'
|
|||
|
|
_COMMA = '\u002C'
|
|||
|
|
_LEFT_PARENTHESIS = '\u0028'
|
|||
|
|
_RIGHT_PARENTHESIS = '\u0029'
|
|||
|
|
|
|||
|
|
|
|||
|
|
_WHITESPACE = {_TAB, _LF, _FF, _CR, _SPACE}
|
|||
|
|
_WHITESPACE_OR_COMMA = _WHITESPACE | {_COMMA}
|
|||
|
|
|
|||
|
|
|
|||
|
|
def parse_srcset(input):
|
|||
|
|
"""
|
|||
|
|
Parse HTML srcset
|
|||
|
|
|
|||
|
|
Based on WhatWG HTML standard § 4.8.4.3.10 Parsing a srcset attribute,
|
|||
|
|
but does not parse or validate descriptors.
|
|||
|
|
|
|||
|
|
https://html.spec.whatwg.org/multipage/images.html#parse-a-srcset-attribute
|
|||
|
|
"""
|
|||
|
|
|
|||
|
|
input_end = len(input)
|
|||
|
|
position = 0
|
|||
|
|
urls = []
|
|||
|
|
|
|||
|
|
while position < input_end:
|
|||
|
|
# 4. Splitting loop: Collect a sequence of code points that are ASCII
|
|||
|
|
# whitespace or U+002C COMMA characters from input given position.
|
|||
|
|
while position < input_end and input[position] in _WHITESPACE_OR_COMMA:
|
|||
|
|
position += 1
|
|||
|
|
|
|||
|
|
# 5. If position is past the end of input, return candidates.
|
|||
|
|
if position >= input_end:
|
|||
|
|
return urls
|
|||
|
|
|
|||
|
|
# 6. Collect a sequence of code points that are not ASCII
|
|||
|
|
# whitespace from input given position, and let that be url.
|
|||
|
|
url_start = position
|
|||
|
|
while position < input_end and input[position] not in _WHITESPACE:
|
|||
|
|
position += 1
|
|||
|
|
url_end = position
|
|||
|
|
|
|||
|
|
# 8, If url ends with U+002C (,), then:
|
|||
|
|
if input[url_end - 1] == _COMMA:
|
|||
|
|
# Remove all trailing U+002C COMMA characters from url.
|
|||
|
|
while url_end > url_start and input[url_end - 1] == _COMMA:
|
|||
|
|
url_end -= 1
|
|||
|
|
else:
|
|||
|
|
# This is a shortened version of 1–4 that simply skips the
|
|||
|
|
# descriptors
|
|||
|
|
while position < input_end:
|
|||
|
|
if input[position] == _LEFT_PARENTHESIS:
|
|||
|
|
# Skip until first closing parenthesis
|
|||
|
|
while (position < input_end and input[position] !=
|
|||
|
|
_RIGHT_PARENTHESIS):
|
|||
|
|
position += 1
|
|||
|
|
elif input[position] == _COMMA:
|
|||
|
|
break
|
|||
|
|
|
|||
|
|
position += 1
|
|||
|
|
|
|||
|
|
# 9-15 is parsing and validation of the descriptors, which we ignore
|
|||
|
|
|
|||
|
|
# If we found an URL
|
|||
|
|
if url_end > url_start:
|
|||
|
|
urls.append(input[url_start:url_end])
|
|||
|
|
|
|||
|
|
return urls
|