-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtasks.py
147 lines (128 loc) · 4.99 KB
/
tasks.py
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
139
140
141
142
143
144
145
146
147
# -*- coding: utf-8 -*-
import os, shlex, shutil, sys, webbrowser
from invoke import task
from invoke.main import program
from pelican import main as pelican_main
from pelican.readers import BaseReader
from pelican.server import ComplexHTTPRequestHandler, RootedHTTPServer
from pelican.settings import DEFAULT_CONFIG, get_settings_from_file
from pelican.utils import slugify
SETTINGS_FILE_BASE = 'pelicanconf.py'
SETTINGS = {}
SETTINGS.update(DEFAULT_CONFIG)
SETTINGS.update(get_settings_from_file(SETTINGS_FILE_BASE))
SETTINGS.update({k:v for k, v in os.environ.items() if k != 'PATH'} ) # useful to override OUTPUT_PATH with an env variable
CONFIG = {
'settings_base': SETTINGS_FILE_BASE,
'settings_publish': 'publishconf.py',
# Output path. Can be absolute or relative to tasks.py. Default: 'output'
'deploy_path': SETTINGS['OUTPUT_PATH'],
# Port for `serve`
'port': 8000,
}
@task
def clean(c):
"Remove generated files"
if os.path.isdir(CONFIG['deploy_path']):
shutil.rmtree(CONFIG['deploy_path'])
os.makedirs(CONFIG['deploy_path'])
@task
def build(c):
"Build local version of site"
cmd = '-s {settings_base} -o {deploy_path}'.format(**CONFIG)
print('build task cmd:', cmd)
pelican_run(cmd)
def src2out(src_file_path):
_, src_file_ext = os.path.splitext(src_file_path)
for cls in BaseReader.__subclasses__():
if cls.enabled and src_file_ext[1:] in cls.file_extensions:
reader = cls(SETTINGS)
break
else:
raise RuntimeError(f'No enabled reader found for extension {src_file_ext}')
_, metadata = reader.read(src_file_path)
joiner = '/'
if metadata.get('status') == 'draft':
joiner = '/drafts/'
elif '/pages/' in src_file_path:
joiner = '/pages/'
slug = metadata.get('slug') or slugify(metadata['title'], regex_subs=SETTINGS['SLUG_REGEX_SUBSTITUTIONS'])
return CONFIG['deploy_path'] + joiner + slug + '.html'
@task
def rebuild(c):
"`build` with the delete switch"
pelican_run('-d -s {settings_base}'.format(**CONFIG))
@task
def regenerate(c):
"Automatically regenerate site upon file modification"
pelican_run('-r -s {settings_base}'.format(**CONFIG))
@task
def serve(c):
"Serve site at http://localhost:$PORT/ (default port is 8000)"
class AddressReuseTCPServer(RootedHTTPServer):
allow_reuse_address = True
server = AddressReuseTCPServer(
CONFIG['deploy_path'],
('', CONFIG['port']),
ComplexHTTPRequestHandler)
sys.stderr.write('Serving on port {port} ...\n'.format(**CONFIG))
server.serve_forever()
@task
def reserve(c):
"`build`, then `serve`"
build(c)
serve(c)
@task
def publish(c):
"Build production version of site"
pelican_run('-s {settings_publish} -o {deploy_path}'.format(**CONFIG))
with open(CONFIG['deploy_path'] + '/tagcloud.html') as tagcloud_file:
tagcloud_lines = extract_lines_between(tagcloud_file.readlines(), '<ul class="mg-tagcloud">', '</ul>')
with open(CONFIG['deploy_path'] + '/pages/bienvenue.html', 'r+') as bienvenue_file:
bienvenue_lines = insert_after_line(bienvenue_file.readlines(), '<!-- tagcloud -->', tagcloud_lines)
bienvenue_file.seek(0)
bienvenue_file.write(''.join(bienvenue_lines))
bienvenue_file.truncate()
def extract_lines_between(in_lines, start_line_content, end_line_content):
out_lines = []
inbetween = False
for line in in_lines:
if start_line_content in line:
inbetween = True
if inbetween:
out_lines.append(line)
if inbetween and end_line_content in line:
break
return out_lines
def insert_after_line(in_lines, target_line_content, add_lines):
out_lines = []
for line in in_lines:
out_lines.append(line)
if target_line_content in line:
out_lines.extend(add_lines)
return out_lines
@task
def livereload(c):
"Automatically reload browser tab upon file modification."
from livereload import Server
build(c)
server = Server()
# Watch the base settings file
server.watch(CONFIG['settings_base'], lambda: build(c))
# Watch content source files
content_file_extensions = ['.md', '.rst']
for extension in content_file_extensions:
content_blob = '{0}/**/*{1}'.format(SETTINGS['PATH'], extension)
server.watch(content_blob, lambda paths: build(c))
# Watch the theme's templates and static assets
theme_path = SETTINGS['THEME']
server.watch('{}/templates/*.html'.format(theme_path), lambda: build(c))
for extension in ('.css', '.js'):
static_file = '{0}/static/**/*{1}'.format(theme_path, extension)
server.watch(static_file, lambda: build(c))
# Serve output path on configured port
webbrowser.open('http://localhost:{}'.format(CONFIG['port']))
server.serve(port=CONFIG['port'], root=CONFIG['deploy_path'])
def pelican_run(cmd):
cmd += ' ' + program.core.remainder # allows to pass-through args to pelican
pelican_main(shlex.split(cmd))