Skip to content

Commit 04fd0ae

Browse files
Secure api_key in zuliprc file using encryption
- Implemented encryption for storing api_key securely in the zuliprc file. - Added decryption logic to retrieve the api_key when needed. - Ensured backward compatibility for existing plaintext api_keys. - Improved security by preventing exposure of sensitive credentials.
1 parent 23a773c commit 04fd0ae

File tree

3 files changed

+148
-4
lines changed

3 files changed

+148
-4
lines changed

tools/encrypt.py

+82
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
1+
#!/usr/bin/python3
2+
3+
import os
4+
import base64
5+
import hashlib
6+
import sys
7+
from cryptography.fernet import Fernet
8+
import argparse
9+
import platform
10+
11+
12+
13+
STORAGE_FILE = os.path.expanduser("~/zuliprc")
14+
15+
16+
17+
parser = argparse.ArgumentParser(description="Demo for optional arguments.")
18+
19+
parser.add_argument("--encrypt", type=str, help="encrypt the given api_key with SHA256 algorithm")
20+
parser.add_argument("--decrypt", help="decrypt the hash using SHA256 algorithm")
21+
22+
args = parser.parse_args()
23+
24+
def in_color(color: str, text: str) -> str:
25+
color_for_str = {
26+
"red": "1",
27+
"green": "2",
28+
"yellow": "3",
29+
"blue": "4",
30+
"purple": "5",
31+
"cyan": "6",
32+
}
33+
# We can use 3 instead of 9 if high-contrast is eg. less compatible?
34+
return f"\033[9{color_for_str[color]}m{text}\033[0m"
35+
36+
37+
def exit_with_error(
38+
error_message: str, *, helper_text: str = "", error_code: int = 1) -> None:
39+
print(in_color("red", error_message))
40+
if helper_text:
41+
print(helper_text)
42+
sys.exit(error_code)
43+
44+
def get_machine_key():
45+
machine_id=''
46+
if platform.system()=="Darwin":
47+
machine_id = os.popen('ioreg -rd1 -c IOPlatformExpertDevice | grep IOPlatformUUID').read().strip()
48+
elif platform.system()=="Linux":
49+
machine_id = os.popen('''blkid | awk -F 'UUID="' '{if (NF>1) print $2}' | cut -d '"' -f1''').read().strip()
50+
51+
key = hashlib.sha256(machine_id.encode()).digest()
52+
53+
return base64.urlsafe_b64encode(key)
54+
55+
def encrypt(api_key):
56+
57+
#getting key from the system
58+
key = get_machine_key()
59+
cipher = Fernet(key)
60+
encrypted_key = cipher.encrypt(api_key.encode())
61+
62+
63+
return encrypted_key.decode()
64+
65+
def decrypt(encrypted_key):
66+
67+
key = get_machine_key()
68+
cipher = Fernet(key)
69+
decrypted_key = cipher.decrypt(encrypted_key)
70+
71+
return decrypted_key.decode()
72+
73+
74+
75+
if args.encrypt :
76+
print(encrypt(args.encrypt))
77+
if args.decrypt:
78+
print(decrypt(args.decrypt))
79+
80+
81+
82+

zulipterminal/cli/run.py

+51-3
Original file line numberDiff line numberDiff line change
@@ -9,9 +9,13 @@
99
import stat
1010
import sys
1111
import traceback
12+
import subprocess
1213
from enum import Enum
13-
from os import path, remove
14+
import re
15+
from os import path , remove
1416
from typing import Dict, List, NamedTuple, Optional, Tuple
17+
import path as pt
18+
1519

1620
import requests
1721
from urwid import display_common, set_encoding
@@ -226,6 +230,11 @@ def parse_args(argv: List[str]) -> argparse.Namespace:
226230
help="profile runtime",
227231
)
228232

233+
parser.add_argument(
234+
"--encrypt",
235+
help="encrypt the hash using SHA256 algorithm",
236+
)
237+
229238
return parser.parse_args(argv)
230239

231240

@@ -312,10 +321,15 @@ def fetch_zuliprc(zuliprc_path: str) -> None:
312321
login_data = get_api_key(realm_url)
313322

314323
preferred_realm_url, login_id, api_key = login_data
324+
325+
326+
command = ['python3', os.path.join(os.path.dirname(__file__), '../../tools/encrypt.py'), '--encrypt', api_key]
327+
encrypted_key = subprocess.run(command, stdout=subprocess.PIPE, text=True)
328+
315329
save_zuliprc_failure = _write_zuliprc(
316330
zuliprc_path,
317331
login_id=login_id,
318-
api_key=api_key,
332+
api_key=encrypted_key.stdout.strip(),
319333
server_url=preferred_realm_url,
320334
)
321335
if not save_zuliprc_failure:
@@ -377,7 +391,6 @@ def parse_zuliprc(zuliprc_str: str) -> Dict[str, SettingData]:
377391
sys.exit(1)
378392

379393
zuliprc = configparser.ConfigParser()
380-
381394
try:
382395
res = zuliprc.read(zuliprc_path)
383396
if len(res) == 0:
@@ -532,7 +545,42 @@ def main(options: Optional[List[str]] = None) -> None:
532545

533546
if args.notify:
534547
zterm["notify"] = SettingData(args.notify, ConfigSource.COMMANDLINE)
548+
549+
if args.encrypt:
550+
config_file = os.path.expanduser(args.encrypt)
551+
config_file = os.path.abspath(config_file)
552+
print(config_file)
553+
api_key,email,site='','',''
554+
if os.path.exists(config_file):
555+
config = configparser.ConfigParser()
556+
config.read(config_file)
557+
api_config = dict(config.items('api'))
558+
api_key = api_config['key']
559+
email = api_config['email']
560+
site = api_config['site']
561+
else:
562+
exit_with_error(f"Could not access zuliprc file at {config_file}")
563+
sys.exit(1)
564+
if len(api_key) >= 64 :
565+
exit_with_error(f"API key is already encrypted")
566+
sys.exit(1)
567+
568+
command = ['python3', os.path.join(os.path.dirname(__file__), '../../tools/encrypt.py'), '--encrypt', api_key]
569+
encrypted_key = subprocess.run(command, stdout=subprocess.PIPE, text=True)
570+
os.remove(config_file)
571+
save_zuliprc_failure = _write_zuliprc(
572+
config_file,
573+
login_id=email,
574+
api_key=encrypted_key.stdout.strip(),
575+
server_url=site,
576+
)
577+
if not save_zuliprc_failure:
578+
print(f"Generated API key saved at {zuliprc_path}")
579+
else:
580+
exit_with_error(save_zuliprc_failure)
581+
sys.exit(0)
535582

583+
536584
valid_remaining_settings = dict(
537585
VALID_BOOLEAN_SETTINGS,
538586
**{"color-depth": COLOR_DEPTH_ARGS_TO_DEPTHS},

zulipterminal/core.py

+15-1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,8 @@
1212
from platform import platform
1313
from types import TracebackType
1414
from typing import Any, Dict, List, Optional, Tuple, Type, Union
15+
import subprocess
16+
import configparser
1517

1618
import pyperclip
1719
import urwid
@@ -98,7 +100,19 @@ def __init__(
98100

99101
self.show_loading()
100102
client_identifier = f"ZulipTerminal/{ZT_VERSION} {platform()}"
101-
self.client = zulip.Client(config_file=config_file, client=client_identifier)
103+
104+
api_key = ''
105+
config_file = os.path.expanduser(config_file)
106+
if os.path.exists(config_file):
107+
config = configparser.ConfigParser()
108+
config.read(config_file)
109+
api_config = dict(config.items('api'))
110+
api_key = api_config['key']
111+
command = ['python3', os.path.join(os.path.dirname(__file__), '../tools/encrypt.py'), '--decrypt', api_key]
112+
113+
decryped_key = subprocess.run(command, stdout=subprocess.PIPE, text=True)
114+
decrypted_key= decryped_key.stdout.strip()
115+
self.client = zulip.Client(config_file=config_file, client=client_identifier,api_key=decrypted_key)
102116
self.model = Model(self)
103117
self.view = View(self)
104118
# Start polling for events after view is rendered.

0 commit comments

Comments
 (0)