2019-04-18 02:00:54 +00:00
|
|
|
__package__ = 'archivebox.legacy.storage'
|
|
|
|
|
2019-04-17 06:25:28 +00:00
|
|
|
import os
|
|
|
|
|
|
|
|
from datetime import datetime
|
|
|
|
from typing import List, Optional
|
|
|
|
|
|
|
|
from ..schema import Link
|
|
|
|
from ..config import (
|
|
|
|
OUTPUT_DIR,
|
|
|
|
TEMPLATES_DIR,
|
|
|
|
VERSION,
|
|
|
|
GIT_SHA,
|
|
|
|
FOOTER_INFO,
|
|
|
|
ARCHIVE_DIR_NAME,
|
2019-04-19 01:09:54 +00:00
|
|
|
HTML_INDEX_FILENAME,
|
2019-04-24 08:10:02 +00:00
|
|
|
STATIC_DIR_NAME,
|
|
|
|
ROBOTS_TXT_FILENAME,
|
|
|
|
FAVICON_FILENAME,
|
2019-04-17 06:25:28 +00:00
|
|
|
)
|
|
|
|
from ..util import (
|
|
|
|
enforce_types,
|
|
|
|
ts_to_date,
|
|
|
|
urlencode,
|
|
|
|
htmlencode,
|
|
|
|
urldecode,
|
|
|
|
wget_output_path,
|
|
|
|
render_template,
|
|
|
|
atomic_write,
|
|
|
|
copy_and_overwrite,
|
|
|
|
)
|
|
|
|
|
|
|
|
join = lambda *paths: os.path.join(*paths)
|
|
|
|
MAIN_INDEX_TEMPLATE = join(TEMPLATES_DIR, 'main_index.html')
|
|
|
|
MAIN_INDEX_ROW_TEMPLATE = join(TEMPLATES_DIR, 'main_index_row.html')
|
|
|
|
LINK_DETAILS_TEMPLATE = join(TEMPLATES_DIR, 'link_details.html')
|
|
|
|
TITLE_LOADING_MSG = 'Not yet archived...'
|
|
|
|
|
|
|
|
|
|
|
|
### Main Links Index
|
|
|
|
|
|
|
|
@enforce_types
|
|
|
|
def write_html_main_index(links: List[Link], out_dir: str=OUTPUT_DIR, finished: bool=False) -> None:
|
|
|
|
"""write the html link index to a given path"""
|
|
|
|
|
2019-04-24 08:10:02 +00:00
|
|
|
copy_and_overwrite(join(TEMPLATES_DIR, FAVICON_FILENAME), join(out_dir, FAVICON_FILENAME))
|
|
|
|
copy_and_overwrite(join(TEMPLATES_DIR, ROBOTS_TXT_FILENAME), join(out_dir, ROBOTS_TXT_FILENAME))
|
|
|
|
copy_and_overwrite(join(TEMPLATES_DIR, STATIC_DIR_NAME), join(out_dir, STATIC_DIR_NAME))
|
2019-04-17 06:25:28 +00:00
|
|
|
|
|
|
|
rendered_html = main_index_template(links, finished=finished)
|
2019-04-19 01:09:54 +00:00
|
|
|
atomic_write(rendered_html, join(out_dir, HTML_INDEX_FILENAME))
|
2019-04-17 06:25:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
@enforce_types
|
|
|
|
def main_index_template(links: List[Link], finished: bool=True) -> str:
|
|
|
|
"""render the template for the entire main index"""
|
|
|
|
|
|
|
|
return render_template(MAIN_INDEX_TEMPLATE, {
|
|
|
|
'version': VERSION,
|
|
|
|
'git_sha': GIT_SHA,
|
|
|
|
'num_links': str(len(links)),
|
|
|
|
'status': 'finished' if finished else 'running',
|
|
|
|
'date_updated': datetime.now().strftime('%Y-%m-%d'),
|
|
|
|
'time_updated': datetime.now().strftime('%Y-%m-%d %H:%M'),
|
|
|
|
'rows': '\n'.join(
|
|
|
|
main_index_row_template(link)
|
|
|
|
for link in links
|
|
|
|
),
|
|
|
|
'footer_info': FOOTER_INFO,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@enforce_types
|
|
|
|
def main_index_row_template(link: Link) -> str:
|
|
|
|
"""render the template for an individual link row of the main index"""
|
|
|
|
|
|
|
|
return render_template(MAIN_INDEX_ROW_TEMPLATE, {
|
|
|
|
**link._asdict(extended=True),
|
|
|
|
|
|
|
|
# before pages are finished archiving, show loading msg instead of title
|
|
|
|
'title': (
|
|
|
|
link.title
|
|
|
|
or (link.base_url if link.is_archived else TITLE_LOADING_MSG)
|
|
|
|
),
|
|
|
|
|
|
|
|
# before pages are finished archiving, show fallback loading favicon
|
|
|
|
'favicon_url': (
|
|
|
|
join(ARCHIVE_DIR_NAME, link.timestamp, 'favicon.ico')
|
|
|
|
# if link['is_archived'] else ''
|
|
|
|
),
|
|
|
|
|
|
|
|
# before pages are finished archiving, show the details page instead
|
|
|
|
'wget_url': urlencode(wget_output_path(link) or 'index.html'),
|
|
|
|
|
|
|
|
# replace commas in tags with spaces, or file extension if it's static
|
|
|
|
'tags': (link.tags or '') + (' {}'.format(link.extension) if link.is_static else ''),
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
### Link Details Index
|
|
|
|
|
|
|
|
@enforce_types
|
|
|
|
def write_html_link_details(link: Link, out_dir: Optional[str]=None) -> None:
|
|
|
|
out_dir = out_dir or link.link_dir
|
|
|
|
|
|
|
|
rendered_html = link_details_template(link)
|
2019-04-19 01:09:54 +00:00
|
|
|
atomic_write(rendered_html, join(out_dir, HTML_INDEX_FILENAME))
|
2019-04-17 06:25:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
@enforce_types
|
|
|
|
def link_details_template(link: Link) -> str:
|
|
|
|
|
|
|
|
link_info = link._asdict(extended=True)
|
|
|
|
|
|
|
|
return render_template(LINK_DETAILS_TEMPLATE, {
|
|
|
|
**link_info,
|
|
|
|
**link_info['canonical'],
|
|
|
|
'title': (
|
|
|
|
link.title
|
|
|
|
or (link.base_url if link.is_archived else TITLE_LOADING_MSG)
|
|
|
|
),
|
|
|
|
'url_str': htmlencode(urldecode(link.base_url)),
|
|
|
|
'archive_url': urlencode(
|
|
|
|
wget_output_path(link)
|
|
|
|
or (link.domain if link.is_archived else 'about:blank')
|
|
|
|
),
|
|
|
|
'extension': link.extension or 'html',
|
|
|
|
'tags': link.tags or 'untagged',
|
|
|
|
'status': 'archived' if link.is_archived else 'not yet archived',
|
|
|
|
'status_color': 'success' if link.is_archived else 'danger',
|
|
|
|
'oldest_archive_date': ts_to_date(link.oldest_archive_date),
|
|
|
|
})
|