2019-04-27 21:26:24 +00:00
|
|
|
"""
|
|
|
|
Everything related to parsing links from input sources.
|
|
|
|
|
|
|
|
For a list of supported services, see the README.md.
|
|
|
|
For examples of supported import formats see tests/.
|
|
|
|
"""
|
|
|
|
|
|
|
|
__package__ = 'archivebox.parsers'
|
|
|
|
|
2019-05-01 03:13:04 +00:00
|
|
|
import re
|
2020-07-29 16:19:06 +00:00
|
|
|
from io import StringIO
|
2019-04-27 21:26:24 +00:00
|
|
|
|
2020-08-18 12:29:05 +00:00
|
|
|
from typing import IO, Tuple, List, Optional
|
2021-04-10 08:19:30 +00:00
|
|
|
from datetime import datetime, timezone
|
2020-09-03 22:26:49 +00:00
|
|
|
from pathlib import Path
|
2019-04-27 21:26:24 +00:00
|
|
|
|
2019-05-01 03:13:04 +00:00
|
|
|
from ..system import atomic_write
|
|
|
|
from ..config import (
|
|
|
|
ANSI,
|
|
|
|
OUTPUT_DIR,
|
|
|
|
SOURCES_DIR_NAME,
|
|
|
|
TIMEOUT,
|
2021-03-31 05:39:01 +00:00
|
|
|
stderr,
|
|
|
|
hint,
|
2019-05-01 03:13:04 +00:00
|
|
|
)
|
2019-04-27 21:26:24 +00:00
|
|
|
from ..util import (
|
2019-05-01 03:13:04 +00:00
|
|
|
basename,
|
2020-08-18 12:23:20 +00:00
|
|
|
htmldecode,
|
2019-05-01 03:13:04 +00:00
|
|
|
download_url,
|
2019-04-27 21:26:24 +00:00
|
|
|
enforce_types,
|
2019-05-01 03:13:04 +00:00
|
|
|
URL_REGEX,
|
2019-04-27 21:26:24 +00:00
|
|
|
)
|
2020-06-26 01:30:29 +00:00
|
|
|
from ..index.schema import Link
|
2020-07-24 17:25:25 +00:00
|
|
|
from ..logging_util import TimedProgress, log_source_saved
|
2020-08-18 12:23:20 +00:00
|
|
|
|
2021-03-31 05:05:49 +00:00
|
|
|
from . import pocket_api
|
|
|
|
from . import wallabag_atom
|
|
|
|
from . import pocket_html
|
|
|
|
from . import pinboard_rss
|
|
|
|
from . import shaarli_rss
|
|
|
|
from . import medium_rss
|
|
|
|
|
|
|
|
from . import netscape_html
|
|
|
|
from . import generic_rss
|
|
|
|
from . import generic_json
|
|
|
|
from . import generic_html
|
|
|
|
from . import generic_txt
|
|
|
|
from . import url_list
|
|
|
|
|
2019-04-27 21:26:24 +00:00
|
|
|
|
2021-03-20 16:38:00 +00:00
|
|
|
PARSERS = {
|
2020-08-18 12:29:05 +00:00
|
|
|
# Specialized parsers
|
2021-03-31 05:05:49 +00:00
|
|
|
pocket_api.KEY: (pocket_api.NAME, pocket_api.PARSER),
|
|
|
|
wallabag_atom.KEY: (wallabag_atom.NAME, wallabag_atom.PARSER),
|
|
|
|
pocket_html.KEY: (pocket_html.NAME, pocket_html.PARSER),
|
|
|
|
pinboard_rss.KEY: (pinboard_rss.NAME, pinboard_rss.PARSER),
|
|
|
|
shaarli_rss.KEY: (shaarli_rss.NAME, shaarli_rss.PARSER),
|
|
|
|
medium_rss.KEY: (medium_rss.NAME, medium_rss.PARSER),
|
|
|
|
|
2020-08-18 12:29:05 +00:00
|
|
|
# General parsers
|
2021-03-31 05:05:49 +00:00
|
|
|
netscape_html.KEY: (netscape_html.NAME, netscape_html.PARSER),
|
|
|
|
generic_rss.KEY: (generic_rss.NAME, generic_rss.PARSER),
|
|
|
|
generic_json.KEY: (generic_json.NAME, generic_json.PARSER),
|
|
|
|
generic_html.KEY: (generic_html.NAME, generic_html.PARSER),
|
2020-08-18 12:29:05 +00:00
|
|
|
|
2021-03-31 05:05:49 +00:00
|
|
|
# Catchall fallback parser
|
|
|
|
generic_txt.KEY: (generic_txt.NAME, generic_txt.PARSER),
|
2021-03-20 16:38:00 +00:00
|
|
|
|
|
|
|
# Explicitly specified parsers
|
2021-03-31 05:05:49 +00:00
|
|
|
url_list.KEY: (url_list.NAME, url_list.PARSER),
|
2021-03-20 16:38:00 +00:00
|
|
|
}
|
2020-08-18 12:29:05 +00:00
|
|
|
|
2020-07-29 16:19:06 +00:00
|
|
|
|
|
|
|
@enforce_types
|
2020-08-18 12:29:05 +00:00
|
|
|
def parse_links_memory(urls: List[str], root_url: Optional[str]=None):
|
2020-07-29 16:19:06 +00:00
|
|
|
"""
|
|
|
|
parse a list of URLS without touching the filesystem
|
|
|
|
"""
|
|
|
|
|
|
|
|
timer = TimedProgress(TIMEOUT * 4)
|
|
|
|
#urls = list(map(lambda x: x + "\n", urls))
|
|
|
|
file = StringIO()
|
|
|
|
file.writelines(urls)
|
|
|
|
file.name = "io_string"
|
2020-08-18 12:29:05 +00:00
|
|
|
links, parser = run_parser_functions(file, timer, root_url=root_url)
|
2020-07-29 16:19:06 +00:00
|
|
|
timer.end()
|
2020-08-18 12:29:05 +00:00
|
|
|
|
|
|
|
if parser is None:
|
|
|
|
return [], 'Failed to parse'
|
|
|
|
return links, parser
|
2020-07-29 16:19:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
@enforce_types
|
2021-03-20 16:38:00 +00:00
|
|
|
def parse_links(source_file: str, root_url: Optional[str]=None, parser: str="auto") -> Tuple[List[Link], str]:
|
2020-07-29 16:19:06 +00:00
|
|
|
"""parse a list of URLs with their metadata from an
|
|
|
|
RSS feed, bookmarks export, or text file
|
|
|
|
"""
|
|
|
|
|
2019-04-27 21:26:24 +00:00
|
|
|
timer = TimedProgress(TIMEOUT * 4)
|
|
|
|
with open(source_file, 'r', encoding='utf-8') as file:
|
2021-03-20 16:38:00 +00:00
|
|
|
links, parser = run_parser_functions(file, timer, root_url=root_url, parser=parser)
|
2019-04-27 21:26:24 +00:00
|
|
|
|
|
|
|
timer.end()
|
2020-08-18 12:29:05 +00:00
|
|
|
if parser is None:
|
|
|
|
return [], 'Failed to parse'
|
|
|
|
return links, parser
|
|
|
|
|
|
|
|
|
2021-03-20 16:38:00 +00:00
|
|
|
def run_parser_functions(to_parse: IO[str], timer, root_url: Optional[str]=None, parser: str="auto") -> Tuple[List[Link], Optional[str]]:
|
2020-08-18 12:29:05 +00:00
|
|
|
most_links: List[Link] = []
|
|
|
|
best_parser_name = None
|
2019-05-01 03:13:04 +00:00
|
|
|
|
2021-03-20 16:38:00 +00:00
|
|
|
if parser != "auto":
|
|
|
|
parser_name, parser_func = PARSERS[parser]
|
|
|
|
parsed_links = list(parser_func(to_parse, root_url=root_url))
|
|
|
|
if not parsed_links:
|
2021-03-31 05:39:01 +00:00
|
|
|
stderr()
|
|
|
|
stderr(f'[X] No links found using {parser_name} parser', color='red')
|
|
|
|
hint('Try a different parser or double check the input?')
|
|
|
|
stderr()
|
2021-03-20 16:38:00 +00:00
|
|
|
timer.end()
|
|
|
|
return parsed_links, parser_name
|
|
|
|
|
|
|
|
for parser_id in PARSERS:
|
|
|
|
parser_name, parser_func = PARSERS[parser_id]
|
2020-07-29 16:19:06 +00:00
|
|
|
try:
|
2020-08-18 12:29:05 +00:00
|
|
|
parsed_links = list(parser_func(to_parse, root_url=root_url))
|
|
|
|
if not parsed_links:
|
2021-03-31 05:39:01 +00:00
|
|
|
raise Exception(f'No links found using {parser_name} parser')
|
2020-08-18 12:29:05 +00:00
|
|
|
|
|
|
|
# print(f'[√] Parser {parser_name} succeeded: {len(parsed_links)} links parsed')
|
|
|
|
if len(parsed_links) > len(most_links):
|
|
|
|
most_links = parsed_links
|
|
|
|
best_parser_name = parser_name
|
|
|
|
|
|
|
|
except Exception as err: # noqa
|
2020-07-29 16:19:06 +00:00
|
|
|
# Parsers are tried one by one down the list, and the first one
|
2021-03-31 05:39:01 +00:00
|
|
|
# that succeeds is used. To debug why a certain parser was not used
|
|
|
|
# due to python error or format incompatibility, uncomment this line:
|
2020-08-18 12:29:05 +00:00
|
|
|
|
2020-07-29 16:19:06 +00:00
|
|
|
# print('[!] Parser {} failed: {} {}'.format(parser_name, err.__class__.__name__, err))
|
|
|
|
# raise
|
2020-08-18 12:29:05 +00:00
|
|
|
pass
|
|
|
|
timer.end()
|
|
|
|
return most_links, best_parser_name
|
2020-07-29 16:19:06 +00:00
|
|
|
|
2019-05-01 03:13:04 +00:00
|
|
|
|
|
|
|
@enforce_types
|
2020-09-03 22:26:49 +00:00
|
|
|
def save_text_as_source(raw_text: str, filename: str='{ts}-stdin.txt', out_dir: Path=OUTPUT_DIR) -> str:
|
2021-04-10 08:19:30 +00:00
|
|
|
ts = str(datetime.now(timezone.utc).timestamp()).split('.', 1)[0]
|
2020-09-30 19:09:34 +00:00
|
|
|
source_path = str(out_dir / SOURCES_DIR_NAME / filename.format(ts=ts))
|
2022-05-10 04:21:39 +00:00
|
|
|
|
|
|
|
referenced_texts = ''
|
|
|
|
|
|
|
|
for entry in raw_text.split():
|
|
|
|
try:
|
2022-11-20 00:51:16 +00:00
|
|
|
if Path(entry).exists():
|
2022-05-10 04:21:39 +00:00
|
|
|
referenced_texts += Path(entry).read_text()
|
|
|
|
except Exception as err:
|
|
|
|
print(err)
|
|
|
|
|
|
|
|
atomic_write(source_path, raw_text + '\n' + referenced_texts)
|
2020-07-13 15:26:30 +00:00
|
|
|
log_source_saved(source_file=source_path)
|
2019-05-01 03:13:04 +00:00
|
|
|
return source_path
|
|
|
|
|
|
|
|
|
|
|
|
@enforce_types
|
2020-09-03 22:26:49 +00:00
|
|
|
def save_file_as_source(path: str, timeout: int=TIMEOUT, filename: str='{ts}-{basename}.txt', out_dir: Path=OUTPUT_DIR) -> str:
|
2019-05-01 03:13:04 +00:00
|
|
|
"""download a given url's content into output/sources/domain-<timestamp>.txt"""
|
2021-04-10 08:19:30 +00:00
|
|
|
ts = str(datetime.now(timezone.utc).timestamp()).split('.', 1)[0]
|
2020-09-30 19:09:34 +00:00
|
|
|
source_path = str(OUTPUT_DIR / SOURCES_DIR_NAME / filename.format(basename=basename(path), ts=ts))
|
2019-05-01 03:13:04 +00:00
|
|
|
|
|
|
|
if any(path.startswith(s) for s in ('http://', 'https://', 'ftp://')):
|
2020-06-26 01:30:29 +00:00
|
|
|
# Source is a URL that needs to be downloaded
|
2020-08-18 12:23:20 +00:00
|
|
|
print(f' > Downloading {path} contents')
|
2019-05-01 03:13:04 +00:00
|
|
|
timer = TimedProgress(timeout, prefix=' ')
|
|
|
|
try:
|
|
|
|
raw_source_text = download_url(path, timeout=timeout)
|
2020-08-18 12:23:20 +00:00
|
|
|
raw_source_text = htmldecode(raw_source_text)
|
2019-05-01 03:13:04 +00:00
|
|
|
timer.end()
|
|
|
|
except Exception as e:
|
|
|
|
timer.end()
|
|
|
|
print('{}[!] Failed to download {}{}\n'.format(
|
|
|
|
ANSI['red'],
|
|
|
|
path,
|
|
|
|
ANSI['reset'],
|
|
|
|
))
|
|
|
|
print(' ', e)
|
2022-05-10 02:56:24 +00:00
|
|
|
raise e
|
2019-05-01 03:13:04 +00:00
|
|
|
|
|
|
|
else:
|
2020-06-26 01:30:29 +00:00
|
|
|
# Source is a path to a local file on the filesystem
|
2019-05-01 03:13:04 +00:00
|
|
|
with open(path, 'r') as f:
|
|
|
|
raw_source_text = f.read()
|
|
|
|
|
2020-06-26 01:30:29 +00:00
|
|
|
atomic_write(source_path, raw_source_text)
|
2019-05-01 03:13:04 +00:00
|
|
|
|
2020-07-13 15:26:30 +00:00
|
|
|
log_source_saved(source_file=source_path)
|
2019-05-01 03:13:04 +00:00
|
|
|
|
|
|
|
return source_path
|
|
|
|
|
|
|
|
|
2021-03-27 07:57:22 +00:00
|
|
|
# Check that plain text regex URL parsing works as expected
|
|
|
|
# this is last-line-of-defense to make sure the URL_REGEX isn't
|
|
|
|
# misbehaving due to some OS-level or environment level quirks (e.g. bad regex lib)
|
|
|
|
# the consequences of bad URL parsing could be disastrous and lead to many
|
|
|
|
# incorrect/badly parsed links being added to the archive, so this is worth the cost of checking
|
|
|
|
_test_url_strs = {
|
|
|
|
'example.com': 0,
|
|
|
|
'/example.com': 0,
|
|
|
|
'//example.com': 0,
|
|
|
|
':/example.com': 0,
|
|
|
|
'://example.com': 0,
|
|
|
|
'htt://example8.com': 0,
|
|
|
|
'/htt://example.com': 0,
|
|
|
|
'https://example': 1,
|
|
|
|
'https://localhost/2345': 1,
|
|
|
|
'https://localhost:1234/123': 1,
|
|
|
|
'://': 0,
|
|
|
|
'https://': 0,
|
|
|
|
'http://': 0,
|
|
|
|
'ftp://': 0,
|
|
|
|
'ftp://example.com': 0,
|
|
|
|
'https://example.com': 1,
|
|
|
|
'https://example.com/': 1,
|
|
|
|
'https://a.example.com': 1,
|
|
|
|
'https://a.example.com/': 1,
|
|
|
|
'https://a.example.com/what/is/happening.html': 1,
|
|
|
|
'https://a.example.com/what/ís/happening.html': 1,
|
|
|
|
'https://a.example.com/what/is/happening.html?what=1&2%20b#höw-about-this=1a': 1,
|
|
|
|
'https://a.example.com/what/is/happéning/?what=1&2%20b#how-aboüt-this=1a': 1,
|
|
|
|
'HTtpS://a.example.com/what/is/happening/?what=1&2%20b#how-about-this=1af&2f%20b': 1,
|
|
|
|
'https://example.com/?what=1#how-about-this=1&2%20baf': 1,
|
|
|
|
'https://example.com?what=1#how-about-this=1&2%20baf': 1,
|
|
|
|
'<test>http://example7.com</test>': 1,
|
2023-08-08 19:09:11 +00:00
|
|
|
'https://<test>': 0,
|
|
|
|
'https://[test]': 0,
|
|
|
|
'http://"test"': 0,
|
|
|
|
'http://\'test\'': 0,
|
2021-03-27 07:57:22 +00:00
|
|
|
'[https://example8.com/what/is/this.php?what=1]': 1,
|
|
|
|
'[and http://example9.com?what=1&other=3#and-thing=2]': 1,
|
|
|
|
'<what>https://example10.com#and-thing=2 "</about>': 1,
|
|
|
|
'abc<this["https://example11.com/what/is#and-thing=2?whoami=23&where=1"]that>def': 1,
|
|
|
|
'sdflkf[what](https://example12.com/who/what.php?whoami=1#whatami=2)?am=hi': 1,
|
|
|
|
'<or>http://examplehttp://15.badc</that>': 2,
|
|
|
|
'https://a.example.com/one.html?url=http://example.com/inside/of/another?=http://': 2,
|
|
|
|
'[https://a.example.com/one.html?url=http://example.com/inside/of/another?=](http://a.example.com)': 3,
|
|
|
|
}
|
|
|
|
for url_str, num_urls in _test_url_strs.items():
|
|
|
|
assert len(re.findall(URL_REGEX, url_str)) == num_urls, (
|
|
|
|
f'{url_str} does not contain {num_urls} urls')
|