2016-02-23 13:23:47 +00:00
|
|
|
#!/usr/bin/env python
|
2016-02-23 23:03:15 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
|
|
|
# Copyright (c) 2016 Ulrik Buchholtz. All rights reserved.
|
|
|
|
# Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
#
|
|
|
|
# Author: Ulrik Buchholtz
|
|
|
|
#
|
2016-02-23 23:02:44 +00:00
|
|
|
# Python 2/3 compatibility
|
|
|
|
from __future__ import print_function
|
2016-02-23 13:23:47 +00:00
|
|
|
|
|
|
|
import argparse
|
|
|
|
import collections
|
|
|
|
import os
|
|
|
|
import sys
|
2016-02-23 19:33:16 +00:00
|
|
|
try:
|
|
|
|
from urllib.request import urlopen
|
|
|
|
from urllib.parse import urlparse
|
|
|
|
except ImportError:
|
|
|
|
from urlparse import urlparse
|
|
|
|
from urllib import urlopen
|
2016-02-23 13:23:47 +00:00
|
|
|
try:
|
|
|
|
import mistune
|
|
|
|
except ImportError:
|
|
|
|
print("Mistune package not found. Install e.g. via `pip install mistune`.")
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(description="Check all *.md files of the current directory's subtree for broken links.")
|
|
|
|
parser.add_argument('--http', help="also check external links (can be slow)", action='store_true')
|
|
|
|
parser.add_argument('--check-missing', help="also find unreferenced lean files", action='store_true')
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
lean_root = os.path.join(os.path.dirname(__file__), os.path.pardir)
|
|
|
|
lean_root = os.path.normpath(lean_root)
|
|
|
|
result = {}
|
|
|
|
|
|
|
|
def check_link(link, root):
|
|
|
|
if link.startswith('http'):
|
|
|
|
if not args.http:
|
|
|
|
return True
|
|
|
|
if link not in result:
|
|
|
|
try:
|
|
|
|
urllib.request.urlopen(link)
|
|
|
|
result[link] = True
|
|
|
|
except:
|
|
|
|
result[link] = False
|
|
|
|
return result[link]
|
|
|
|
else:
|
|
|
|
if link.startswith('/'):
|
|
|
|
# project root-relative link
|
|
|
|
path = lean_root + link
|
|
|
|
else:
|
|
|
|
path = os.path.join(root, link)
|
|
|
|
|
|
|
|
path = os.path.normpath(path) # should make it work on Windows
|
|
|
|
|
|
|
|
result[path] = os.path.exists(path)
|
|
|
|
return result[path]
|
|
|
|
|
|
|
|
# check all .md files
|
|
|
|
for root, _, files in os.walk('.'):
|
|
|
|
for f in files:
|
|
|
|
if not f.endswith('.md'):
|
|
|
|
continue
|
|
|
|
|
|
|
|
path = os.path.join(root, f)
|
|
|
|
|
|
|
|
class CheckLinks(mistune.Renderer):
|
|
|
|
def link(self, link, title, content):
|
|
|
|
if not check_link(link, root):
|
|
|
|
print("Broken link", link, "in file", path)
|
|
|
|
|
|
|
|
mistune.Markdown(renderer=CheckLinks())(open(path).read())
|
|
|
|
|
|
|
|
if args.check_missing:
|
|
|
|
# check all .(h)lean files
|
|
|
|
for root, _, files in os.walk('.'):
|
|
|
|
for f in files:
|
|
|
|
path = os.path.normpath(os.path.join(root, f))
|
|
|
|
if (path.endswith('.lean') or path.endswith('.hlean')) and path not in result:
|
|
|
|
result[path] = False
|
|
|
|
print("Missing file", path)
|
|
|
|
|
|
|
|
if not all(result.values()):
|
|
|
|
sys.exit(1)
|