Files: 3e4995ca4388b536ae7820c7655650720bb45e8c / scuttleblog.py
5060 bytesRaw
1 | #!/usr/bin/python3 |
2 | |
3 | __copyright__ = """ |
4 | |
5 | Scuttleblog |
6 | |
7 | Copyright (C) 2017 Greg K Nicholson |
8 | |
9 | This program is free software: you can redistribute it and/or modify |
10 | it under the terms of the GNU Affero General Public License as published by |
11 | the Free Software Foundation, either version 3 of the License, or |
12 | (at your option) any later version. |
13 | |
14 | This program is distributed in the hope that it will be useful, |
15 | but WITHOUT ANY WARRANTY; without even the implied warranty of |
16 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
17 | GNU Affero General Public License for more details. |
18 | |
19 | You should have received a copy of the GNU Affero General Public License |
20 | along with this program. If not, see <https://www.gnu.org/licenses/>. |
21 | |
22 | """ |
23 | |
24 | import datetime |
25 | import json |
26 | import os |
27 | import re |
28 | import time |
29 | import subprocess |
30 | |
31 | import conf |
32 | |
33 | scuttleblog_dir = os.path.dirname(__file__) |
34 | |
35 | def run_sbot(): |
36 | for attempt in range(100): |
37 | try: |
38 | subprocess.check_output(['sbot', 'status']) |
39 | except: |
40 | pid = subprocess.Popen(['sbot', 'server']).pid |
41 | time.sleep(1) |
42 | else: |
43 | break |
44 | |
45 | def get_user_posts(ssb_userid): |
46 | user_posts_args = ['sbot', 'createUserStream', '--id', ssb_userid] |
47 | json_posts_args = ['json', '--group', '-c', 'this.value.content.type == "post"', '-c', 'this.value.content.root == null'] |
48 | user_posts_stream = subprocess.Popen(user_posts_args, stdout = subprocess.PIPE) |
49 | user_posts_json = subprocess.check_output(json_posts_args, stdin = user_posts_stream.stdout) |
50 | user_posts = json.loads(user_posts_json) |
51 | return user_posts |
52 | |
53 | def define_post_text(text): |
54 | text = text.strip() |
55 | title = text.splitlines()[0] |
56 | if re.match('#', text): |
57 | # The first line is a heading. Use it as the title and strip it from the body. |
58 | body = ''.join(text.splitlines(keepends=True)[1:]).strip() \ |
59 | if len(text.splitlines()) > 1 \ |
60 | else '' |
61 | else: |
62 | # Truncate the first line and use it as the title. Keep everything in the body. |
63 | maxlength = 100 |
64 | ellipsis = '...' |
65 | title = title if len(title) <= maxlength else title[:maxlength].rsplit(' ', 1)[0] + ellipsis |
66 | body = text |
67 | title = re.sub('^#+\s+', '', title) |
68 | return {'title': title, 'body': body} |
69 | |
70 | def build_post_structure(p): |
71 | post = {} |
72 | post['frontmatter'] = {} |
73 | post['frontmatter']['key'] = p['key'] |
74 | post['frontmatter']['title'] = define_post_text(p['value']['content']['text'])['title'] |
75 | post['frontmatter']['date'] = datetime.datetime.fromtimestamp(int(p['value']['timestamp'] / 1000), datetime.timezone.utc).strftime('%Y-%m-%dT%H:%M:%SZ') |
76 | post['frontmatter']['sequence'] = p['value']['sequence'] |
77 | post['body'] = define_post_text(p['value']['content']['text'])['body'] |
78 | return (post) |
79 | |
80 | def format_post_file(post): |
81 | content = '' |
82 | content += str(json.dumps(post['frontmatter'], indent = 4)) |
83 | content += '\n\n' |
84 | content += str(post['body']) |
85 | return content |
86 | |
87 | def define_post_filename(post): |
88 | folder = 'hugo/content/posts' |
89 | #slug = post['key'].replace('+','-').replace('/','_').replace('=.sha256','') |
90 | slug = str(post['frontmatter']['sequence']) |
91 | filetype = 'md' |
92 | return folder + '/' + slug + '.' + filetype |
93 | |
94 | def write_post_file(post): |
95 | os.makedirs(os.path.dirname(define_post_filename(post)), exist_ok=True) |
96 | with open(define_post_filename(post), 'w') as f: |
97 | f.write (format_post_file(post)) |
98 | |
99 | def write_posts_from_user(ssb_userid): |
100 | posts = get_user_posts(ssb_userid) |
101 | for post in posts: |
102 | write_post_file(build_post_structure(post)) |
103 | |
104 | |
105 | def get_user_metadata(ssb_userid): |
106 | user_metadata_args = ['sbot', 'links', '--source', ssb_userid, '--dest', ssb_userid, '--rel', 'about', '--values'] |
107 | json_metadata_args = ['json', '--deep-merge', '-c', 'this.value.content.type == "about"'] |
108 | user_metadata_stream = subprocess.Popen(user_metadata_args, stdout = subprocess.PIPE) |
109 | user_metadata_json = subprocess.check_output(json_metadata_args, stdin = user_metadata_stream.stdout) |
110 | user_metadata = json.loads(user_metadata_json) |
111 | return user_metadata |
112 | |
113 | def check_hugo_theme(theme, theme_clone_url): |
114 | hugo_theme_dir = os.path.join(scuttleblog_dir, 'hugo', 'themes', theme) |
115 | if not os.path.isdir(hugo_theme_dir): |
116 | subprocess.run(['git', 'submodule', 'add', '-f', theme_clone_url, hugo_theme_dir]) |
117 | |
118 | def build_hugo_config(m): |
119 | hugo_config = {} |
120 | hugo_config['baseurl'] = conf.hugo_baseurl |
121 | hugo_config['theme'] = conf.hugo_theme |
122 | if 'name' in m['value']['content']: |
123 | hugo_config['title'] = m['value']['content']['name'] |
124 | hugo_config['params'] = {} |
125 | if 'description' in m['value']['content']: |
126 | hugo_config['params']['subtitle'] = m['value']['content']['description'].replace('\n', ' ') |
127 | return (hugo_config) |
128 | |
129 | def write_hugo_config_from_user(ssb_userid): |
130 | metadata = get_user_metadata(ssb_userid) |
131 | for file in ['hugo/config.toml', 'hugo/config.yaml']: |
132 | if os.path.exists(file): |
133 | os.remove(file) |
134 | with open('hugo/config.json', 'w') as f: |
135 | f.write (json.dumps(build_hugo_config(metadata), indent = 4)) |
136 | |
137 | def run_hugo(): |
138 | subprocess.run(['hugo', '-s', 'hugo']) |
139 | |
140 | run_sbot() |
141 | write_posts_from_user(conf.ssb_userid) |
142 | write_hugo_config_from_user(conf.ssb_userid) |
143 | check_hugo_theme(conf.hugo_theme, conf.hugo_theme_clone_url) |
144 | run_hugo() |
145 | |
146 |
Built with git-ssb-web