-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmarkdown_tree
executable file
·138 lines (108 loc) · 4.14 KB
/
markdown_tree
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
#!/usr/bin/env python3
from urllib.parse import quote
from pathlib import Path
import json
import click
MARKDOWN_HEADER = """
# Contents
"""
MARKDOWN_PREFIX = [
"## {}",
"- {}",
]
MARKDOWN_FOOTER = """
-----
Generated by [markdown-tree](https://pypi.org/project/markdown-tree/)
""".strip()
IGNORED_FILES = [".*"]
class MarkdownTreeGenerator(object):
def __init__(self, dir, ignore, header, footer, emoji, prefixes):
self.dir = dir
self.ignore = ignore
self.header = header
self.footer = footer
self.prefixes = prefixes
self.emoji = emoji
def generate(self):
if self.header:
yield self.header
for line in self.walk(Path(self.dir), 0):
yield line
if self.footer:
yield self.footer
def walk(self, dir, level):
for entry in sorted(self.iterdir(dir)):
content = self.link(entry.name, quote(str(entry)))
emoji = self.get_emoji(entry)
if emoji:
content = '{} {}'.format(emoji, content)
line = self.markdown_prefix(level).format(content)
if entry.is_file():
yield line
if entry.is_dir():
if list(self.iterdir(entry)):
yield line
for line in self.walk(entry, level + 1):
yield line
def iterdir(self, dir):
for entry in sorted(dir.iterdir()):
if self.check_against_filters(entry):
yield entry
def check_against_filters(self, entry):
for i in self.ignore:
if i.endswith('/'):
if entry.is_dir() and entry.match(i):
return False
else:
if entry.match(i):
return False
return True
def markdown_prefix(self, i):
if i < len(self.prefixes):
return self.prefixes[i]
return ' ' * (i - len(self.prefixes) + 1) + self.prefixes[-1]
def link(self, text, href):
return '[{}]({})'.format(text, href)
def get_emoji(self, entry):
if not self.emoji:
return None
if entry.is_file():
return '📄'
if entry.is_dir():
return '📁'
return None
@click.command()
@click.argument("directory", type=click.Path(exists=True), default=".")
@click.option("--ignore", type=str, default=','.join(IGNORED_FILES), help="Ignored files or dirs, comma separated, default=.*")
@click.option("--header", type=str, default=MARKDOWN_HEADER, help="Header")
@click.option("--footer", type=str, default=MARKDOWN_FOOTER, help="Footer")
@click.option("--emoji/--no-emoji", default=False, help="Enable file/folder emoji")
@click.option("--prefixes", type=str, default=','.join(MARKDOWN_PREFIX), help="Prefixes, default='## {},- {}', which means, top level entries are <h2> and the rest are <ul>")
@click.option("-c", "--config", type=click.Path(exists=False), default=".mdtreeconfig.json", help="Load all configs from a json file if exists, default=.mdtreeconfig.json")
@click.option("-o", "--output", type=click.Path(exists=False), default=None, help="Output filename, default is stardard output")
def generate_markdown_tree(directory, ignore="", header="", footer="", emoji=False, prefixes="", config="", output=None):
directory = Path(directory)
options = {}
if config and (directory / config).exists():
options = json.loads((directory / config).read_text())
if 'ignore' not in options:
options['ignore'] = ignore.split(',')
if 'header' not in options:
options['header'] = header
if 'footer' not in options:
options['footer'] = footer
if 'emoji' not in options:
options['emoji'] = emoji
if 'prefixes' not in options:
options['prefixes'] = prefixes.split(',')
if 'output' not in options:
options['output'] = output
output = options.pop('output')
gen = MarkdownTreeGenerator(dir=directory, **options)
text = '\n'.join(gen.generate())
if output:
Path(output).write_text(text)
else:
print(text)
if __name__ == "__main__":
generate_markdown_tree()