-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsetup_murmeli.py
307 lines (277 loc) · 13.6 KB
/
setup_murmeli.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
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
'''Command-line tools to setup the murmeli system.
Without dependencies to Qt, it could also be used to
setup a robot system on a screenless pi, for example.'''
import os
import sys
import pkg_resources as pkgs
from murmeli.system import System
from murmeli.config import Config
from murmeli.cryptoclient import CryptoClient
from murmeli.i18n import I18nManager
from murmeli.torclient import TorClient
from murmeli.supersimpledb import MurmeliDb
def check_dependencies():
'''Check whether all required software is installed'''
# Check python
python_version = (sys.version_info.major, sys.version_info.minor,
sys.version_info.micro)
print("Python verson: %d.%d.%d" % python_version)
if python_version < (3, 4):
print("Murmeli needs at least version 3.4 to run. Ending.")
sys.exit(1)
# Try to import python-gnupg, just to see if it's there
try:
from gnupg import GPG
if GPG.verify:
print("Found GnuPG")
except ImportError:
print("Murmeli needs python-gnupg to run. Ending.")
sys.exit(1)
try:
print("Found GnuPG version: %s" % pkgs.get_distribution("python-gnupg").version)
except pkgs.ResolutionError:
print("Could not find GnuPG version")
def check_config(system):
'''Look for a config file, and load it if possible'''
conf = Config(system)
system.add_component(conf)
system.invoke_call(System.COMPNAME_I18N, "set_language")
print(make_heading(system, "startupwizard.title"))
# Select which language to use
selected_language = ask_question(system, "setup.language", ["setup.lang.en", "setup.lang.de"])
check_abort(selected_language, system)
lang = "de" if selected_language == "2" else "en"
conf.set_property(conf.KEY_LANGUAGE, lang)
system.invoke_call(System.COMPNAME_I18N, "set_language")
print(get_text(system, "setup.languageselected"))
if not conf.from_file:
conf.save()
print(get_text(system, "setup.configsaved"))
def check_keyring(system):
'''Given the data directory, is there a keyring and are there keys in it?'''
crypto = CryptoClient(system)
system.add_component(crypto)
gpg_version = None
while not gpg_version:
gpg_version = crypto.get_gpg_version()
if gpg_version:
print(get_text(system, "setup.foundgpgversion") % gpg_version)
else:
print(get_text(system, "setup.entergpgpath"))
gpg_path = input("? ")
if gpg_path in ["", "q"]:
check_abort("q", system)
system.invoke_call(System.COMPNAME_CONFIG, "set_property",
key=Config.KEY_GPG_EXE, value=gpg_path)
print(get_text(system, "setup.foundkeyring" if crypto.found_keyring() else "setup.nokeyring"))
print(get_text(system, "setup.foundkeys") % (crypto.get_num_keys(private_keys=True),
crypto.get_num_keys(public_keys=True)))
if crypto.get_num_keys(private_keys=True) < 1:
gen_pair = ask_question(system, "setup.genkeypair", ["setup.genkeypair.rsa"])
check_abort(gen_pair, system)
generate_keypair(crypto, system)
def generate_keypair(crypto, system):
'''Generate a new private/public key pair using the given data'''
key_name = None
while not key_name:
key_name = input(get_text(system, "setup.genkeypair.name") + ": ")
key_email = input(get_text(system, "setup.genkeypair.email") + ": ") or "[email protected]"
key_comment = input(get_text(system, "setup.genkeypair.comment") + ": ")
# Pass these fields to gpg
print(get_text(system, "setup.genkeypair.pleasewait"))
result = crypto.generate_key_pair(key_name, key_email, key_comment)
print(get_text(system, "setup.genkeypair.complete"))
return result
def check_data_path(system):
'''Check if the data path is configured and if it exists or not'''
data_path = system.invoke_call(System.COMPNAME_CONFIG, "get_data_dir")
if data_path:
print(get_text(system, "setup.datadir"), ":", data_path)
selected_data_path = None
while not selected_data_path:
selected_data_path = input(get_text(system, "setup.datadir") + "? ")
if selected_data_path:
data_path = selected_data_path
else:
selected_data_path = data_path
if data_path:
data_path = os.path.abspath(os.path.expanduser(data_path))
if not os.path.exists(data_path):
# Confirm before creation
print(get_text(system, "setup.datadir"), ":", data_path)
create_dir = ask_question(system, "setup.createdatadir",
["setup.createdir.create", "setup.createddir.cancel"])
check_abort(create_dir, system)
if create_dir == "2":
selected_data_path = None
system.invoke_call(System.COMPNAME_CONFIG, "set_property",
key=Config.KEY_DATA_DIR, value=data_path)
if not os.path.exists(data_path):
print(get_text(system, "setup.datadir.creating"))
os.makedirs(system.invoke_call(System.COMPNAME_CONFIG, "get_database_dir"), exist_ok=True)
os.makedirs(system.invoke_call(System.COMPNAME_CONFIG, "get_web_cache_dir"), exist_ok=True)
os.makedirs(system.invoke_call(System.COMPNAME_CONFIG, "get_keyring_dir"), exist_ok=True)
os.makedirs(system.invoke_call(System.COMPNAME_CONFIG, "get_tor_dir"), exist_ok=True)
def check_tor(system):
'''Try to start tor, and get the tor id if possible'''
print(get_text(system, "setup.entertorpath"))
tor_path = input("? ") or "tor"
# Save tor path in config
system.invoke_call(System.COMPNAME_CONFIG, "set_property",
key=Config.KEY_TOR_EXE, value=tor_path)
tor_dir = system.invoke_call(System.COMPNAME_CONFIG, "get_tor_dir")
tor_client = TorClient(None, tor_dir, tor_path)
started, torid = tor_client.ignite_to_get_tor_id()
if not started and not torid:
print(get_text(system, "setup.startingtorfailed"))
elif torid:
print(get_text(system, "setup.foundtorid") % torid)
return torid
def select_keypair(system):
'''If there is more than one keypair, select which one to use'''
num_private_keys = system.invoke_call(System.COMPNAME_CRYPTO, "get_num_keys",
private_keys=True)
if num_private_keys < 1:
print("The keyring has no private keys, wasn't one already generated?")
return None
private_keys = system.invoke_call(System.COMPNAME_CRYPTO, "get_keys", private_keys=True)
key_index = "1"
if num_private_keys > 1:
# Need to choose which key to use
answers = []
for key in private_keys:
name = key['uids']
name = str(name[0]) if isinstance(name, list) else name
answers.append("%s (%s)" % (key['keyid'], name))
question = get_text(system, "setup.selectprivatekey")
key_index = ask_question_raw(system, question, answers)
check_abort(key_index, system)
return private_keys[int(key_index) - 1].get('keyid')
def select_robot_status(system, private_keyid):
'''Specify whether we're setting up a robot system or a real system'''
system_type = ask_question(system, "setup.realorrobot",
["setup.system.real", "setup.system.robot", "setup.system.parrot"])
check_abort(system_type, system)
# print("Selected system type:", system_type)
data_path = system.invoke_call(System.COMPNAME_CONFIG, "get_data_dir")
if system_type == "1":
# Real system, not a robot
system.invoke_call(System.COMPNAME_CONFIG, "set_property",
key=Config.KEY_ROBOT_OWNER_KEY, value="")
# Check whether to export the public key or not
export_index = ask_question(system, "setup.exportpublickey", ["setup.no", "setup.yes"])
check_abort(export_index, system)
if export_index == "2":
keyfile_name = private_keyid + ".key"
with open(os.path.join(data_path, keyfile_name), "w") as keyfile:
keyfile.write(system.invoke_call(System.COMPNAME_CRYPTO, "get_public_key",
key_id=private_keyid))
print(get_text(system, "setup.publickeyexported") % keyfile_name)
elif system_type in ["2", "3"]:
own_name = "Parrot" if system_type == "3" else "Robot"
return (setup_robot_system(system, data_path, private_keyid), own_name)
return (None, None)
def setup_robot_system(system, data_path, private_keyid):
'''Choose which key to use for the robot system's owner and save this choice'''
while True:
# Collect the public keys from keyring, but not own one
public_keys = system.invoke_call(System.COMPNAME_CRYPTO, "get_keys",
public_keys=True)
public_keys = [key for key in public_keys if key['keyid'] != private_keyid]
print("Keyring has %d public keys: %s" % (len(public_keys), public_keys))
# Also find any .key files in data directory
keyfiles = [file for file in os.listdir(data_path) if file.endswith(".key")
and os.path.isfile(os.path.join(data_path, file))]
# Assemble list of available public keys
key_list = ["%s (%s)" % (key['keyid'], key['uids'][0]) for key in public_keys] + \
[os.path.basename(file) for file in keyfiles] + \
[get_text(system, "setup.refreshkeylist")]
files_to_load = [None for _ in public_keys] + keyfiles
print(key_list, files_to_load)
# Ask question which one to take (& mention that keys can be copied to datadir)
owner_key = ask_question_raw(system, get_text(system, "setup.selectrobotownerkey"),
key_list)
check_abort(owner_key, system)
refresh_index = len(key_list)
if owner_key != str(refresh_index):
key_index = int(owner_key) - 1
file_to_load = files_to_load[key_index]
if file_to_load:
# Selected key is in a file to load, so import into the keyring
with open(os.path.join(data_path, file_to_load), "r") as keyfile:
key = "".join(keyfile.readlines())
key_id = system.invoke_call(System.COMPNAME_CRYPTO,
"import_public_key", strkey=key)
else:
key_id = public_keys[key_index]['keyid']
# Store owner key id in config
system.invoke_call(System.COMPNAME_CONFIG, "set_property",
key=Config.KEY_ROBOT_OWNER_KEY, value=key_id)
return key_id
def setup_database(system, torid, private_keyid, own_name=None):
'''Setup database and store private key'''
print("Storing database: tor id='%s', key id='%s'" % (torid, private_keyid))
name = (own_name + torid[:12]) if own_name else torid
db_filename = system.invoke_call(System.COMPNAME_CONFIG, "get_ss_database_file")
ssdb = MurmeliDb(None, db_filename)
ssdb.add_or_update_profile({"torid":torid, "keyid":private_keyid, "status":"self",
"ownprofile":True, "name":name})
ssdb.save_to_file()
def ask_question(system, question_key, answer_keys):
'''Use the console to ask the user a question and collect the answer, using token keys'''
return ask_question_raw(system, get_text(system, question_key),
[get_text(system, key) for key in answer_keys])
def ask_question_raw(system, question_text, answer_texts):
'''Use the console to ask the user a question and collect the answer, using strings'''
# Loop until a valid answer is given
while True:
try:
print("\n" + question_text, get_text(system, "setup.qtoquit"))
allowed_answers = ["q"]
for index, txt in enumerate(answer_texts):
print("%d : %s" % (index + 1, txt))
allowed_answers += str(index + 1)
answer = input("? ")
if answer in allowed_answers:
return answer
except KeyboardInterrupt:
return "q"
def check_abort(answer, system):
'''Check the returned answer and abort if requested'''
if answer == 'q':
print(get_text(system, "setup.abort"))
sys.exit(1)
def make_heading(system, key):
'''Make a heading string for printing to the console'''
title = get_text(system, key)
return "\n".join(["", title, "-" * len(title)])
def get_text(system, key):
'''Convenience methods for getting texts from the system's i18n'''
text = system.invoke_call(System.COMPNAME_I18N, "get_text", key=key)
return text.replace("\\n", "\n") if text else ""
def setup_murmeli():
'''Run through all the checks and setup the config, keyring and database'''
check_dependencies()
system = System()
i18n = I18nManager(system)
system.add_component(i18n)
check_config(system)
check_data_path(system)
tor_id = check_tor(system)
check_abort(tor_id or 'q', system)
check_keyring(system)
# Select which keypair to use
private_keyid = select_keypair(system)
if not private_keyid:
print("ERROR: Failed to generate keypair. Aborting.")
return
print("Selected key '%s'" % private_keyid)
# Define robot parameters
owner_keyid, own_name = select_robot_status(system, private_keyid)
print("Owner keyid '%s'" % owner_keyid)
# Setup database and store private key
setup_database(system, tor_id, private_keyid, own_name)
# Save config
system.invoke_call(System.COMPNAME_CONFIG, "save")
if __name__ == "__main__":
setup_murmeli()