forked from richardpenman/browsercookie
-
Notifications
You must be signed in to change notification settings - Fork 0
/
__init__.py
451 lines (380 loc) · 17.8 KB
/
__init__.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
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
# -*- coding: utf-8 -*-
__doc__ = 'Load browser cookies into a cookiejar'
import os
import sys
import time
import glob
import base64
from win32 import win32crypt #pywin32
from Crypto.Cipher import AES
from Crypto.Util.Padding import unpad
try:
import cookielib
except ImportError:
import http.cookiejar as cookielib
from contextlib import contextmanager
import tempfile
try:
import json
except ImportError:
import simplejson as json
try:
import ConfigParser as configparser
except ImportError:
import configparser
try:
# should use Cryptodome in windows instead of Crypto
# otherwise will raise an import error
from Cryptodome.Protocol.KDF import PBKDF2
from Cryptodome.Cipher import AES
except ImportError:
from Crypto.Protocol.KDF import PBKDF2
from Crypto.Cipher import AES
try:
# should use pysqlite2 to read the cookies.sqlite on Windows
# otherwise will raise the "sqlite3.DatabaseError: file is encrypted or is not a database" exception
from pysqlite2 import dbapi2 as sqlite3
except ImportError:
import sqlite3
if sys.platform == 'darwin': # darwin is OSX
from struct import unpack
try:
from StringIO import StringIO # only works for python2
except ImportError:
from io import BytesIO as StringIO # only works for python3
import lz4.block
import keyring
class BrowserCookieError(Exception):
pass
@contextmanager
def create_local_copy(cookie_file):
"""
Make a local copy of the sqlite cookie database and return the new filename.
This is necessary in case this database is still being written to while the user browses
to avoid sqlite locking errors.
"""
# check if cookie file exists
if os.path.exists(cookie_file):
# copy to random name in tmp folder
tmp_cookie_file = tempfile.NamedTemporaryFile(suffix='.sqlite').name
open(tmp_cookie_file, 'wb').write(open(cookie_file, 'rb').read())
yield tmp_cookie_file
else:
raise BrowserCookieError('Can not find cookie file at: ' + cookie_file)
os.remove(tmp_cookie_file)
class BrowserCookieLoader(object):
def __init__(self, cookie_files=None):
cookie_files = cookie_files or self.find_cookie_files()
self.cookie_files = list(cookie_files)
def find_cookie_files(self):
'''Return a list of cookie file locations valid for this loader'''
raise NotImplementedError
def get_cookies(self):
'''Return all cookies (May include duplicates from different sources)'''
raise NotImplementedError
def load(self):
'''Load cookies into a cookiejar'''
cookie_jar = cookielib.CookieJar()
for cookie in self.get_cookies():
cookie_jar.set_cookie(cookie)
return cookie_jar
class Chrome(BrowserCookieLoader):
def __str__(self):
return 'chrome'
def find_cookie_files(self):
for pattern in [
os.path.expanduser('~/Library/Application Support/Google/Chrome/Default/Cookies'),
os.path.expanduser('~/Library/Application Support/Google/Chrome/Profile */Cookies'),
os.path.expanduser('~/Library/Application Support/Vivaldi/Default/Cookies'),
os.path.expanduser('~/Library/Application Support/Vivaldi/Profile */Cookies'),
os.path.expanduser('~/.config/chromium/Default/Cookies'),
os.path.expanduser('~/.config/chromium/Profile */Cookies'),
os.path.expanduser('~/.config/google-chrome/Default/Cookies'),
os.path.expanduser('~/.config/google-chrome/Profile */Cookies'),
os.path.expanduser('~/.config/vivaldi/Default/Cookies'),
os.path.expanduser('~/.config/vivaldi/Profile */Cookies'),
os.path.join(os.getenv('APPDATA', ''), r'..\Local\Google\Chrome\User Data\Default\Cookies'),
os.path.join(os.getenv('APPDATA', ''), r'..\Local\Google\Chrome\User Data\Profile *\Cookies'),
os.path.join(os.getenv('APPDATA', ''), r'..\Local\Vivaldi\User Data\Default\Cookies'),
os.path.join(os.getenv('APPDATA', ''), r'..\Local\Vivaldi\User Data\Profile *\Cookies'),
]:
for result in glob.glob(pattern):
yield result
def get_cookies(self):
salt = b'saltysalt'
length = 16
if sys.platform == 'darwin':
# running Chrome on OSX
my_pass = keyring.get_password('Chrome Safe Storage', 'Chrome')
my_pass = my_pass.encode('utf8')
iterations = 1003
key = PBKDF2(my_pass, salt, length, iterations)
elif sys.platform.startswith('linux'):
# running Chrome on Linux
my_pass = 'peanuts'.encode('utf8')
iterations = 1
key = PBKDF2(my_pass, salt, length, iterations)
elif sys.platform == 'win32':
path = r'%LocalAppData%\Google\Chrome\User Data\Local State'
path = os.path.expandvars(path)
with open(path, 'r') as file:
encrypted_key = json.loads(file.read())['os_crypt']['encrypted_key']
encrypted_key = base64.b64decode(encrypted_key) # Base64 decoding
encrypted_key = encrypted_key[5:] # Remove DPAPI
key = win32crypt.CryptUnprotectData(encrypted_key, None, None, None, 0)[1] # Decrypt key
else:
raise BrowserCookieError('Unsupported operating system: ' + sys.platform)
for cookie_file in self.cookie_files:
with create_local_copy(cookie_file) as tmp_cookie_file:
con = sqlite3.connect(tmp_cookie_file)
cur = con.cursor()
cur.execute('SELECT value FROM meta WHERE key = "version";')
version = int(cur.fetchone()[0])
query = 'SELECT host_key, path, is_secure, expires_utc, name, value, encrypted_value FROM cookies;'
if version < 10:
query = query.replace('is_', '')
cur.execute(query)
for item in cur.fetchall():
host, path, secure, expires, name = item[:5]
value = self._decrypt(item[5], item[6], item[4], item[1], key=key)
yield create_cookie(host, path, secure, expires, name, value)
con.close()
def _decrypt(self, value, encrypted_value, cookiename, sitename, key):
"""Decrypt encoded cookies
"""
if (sys.platform == 'darwin') or sys.platform.startswith('linux'):
if value or (encrypted_value[:3] != b'v10'):
return value
# Encrypted cookies should be prefixed with 'v10' according to the
# Chromium code. Strip it off.
encrypted_value = encrypted_value[3:]
# Strip padding by taking off number indicated by padding
# eg if last is '\x0e' then ord('\x0e') == 14, so take off 14.
def clean(x):
last = x[-1]
if isinstance(last, int):
return x[:-last].decode('utf8')
else:
return x[:-ord(last)].decode('utf8')
iv = b' ' * 16
cipher = AES.new(key, AES.MODE_CBC, IV=iv)
decrypted = cipher.decrypt(encrypted_value)
return clean(decrypted)
else:
# Must be win32 (on win32, all chrome cookies are encrypted)
plaintext = ""
if encrypted_value[:3] == b'v10':
try:
data = encrypted_value # the encrypted cookie
nonce = data[3:3 + 12]
ciphertext = data[3 + 12:-16]
tag = data[-16:]
cipher = AES.new(key, AES.MODE_GCM, nonce=nonce)
plaintext = cipher.decrypt_and_verify(ciphertext, tag).decode("utf-8") # the decrypted cookie
except:
raise BrowserCookieError("Error decrypting V80+ cookie: " + str(cookiename) + " from site " + str(sitename))
else:
try:
plaintext = win32crypt.CryptUnprotectData(encrypted_value, None, None, None, 0)[1].decode("utf-8")
except:
raise BrowserCookieError("Error decrypting cookie: " + str(cookiename) + " from site " + str(sitename))
return plaintext
class Firefox(BrowserCookieLoader):
def __str__(self):
return 'firefox'
def parse_profile(self, profile):
cp = configparser.ConfigParser()
cp.read(profile)
path = None
for section in cp.sections():
try:
if cp.getboolean(section, 'IsRelative'):
path = os.path.dirname(profile) + '/' + cp.get(section, 'Path')
else:
path = cp.get(section, 'Path')
if cp.has_option(section, 'Default'):
return os.path.abspath(os.path.expanduser(path))
except configparser.NoOptionError:
pass
if path:
return os.path.abspath(os.path.expanduser(path))
raise BrowserCookieError('No default Firefox profile found')
def find_default_profile(self):
if sys.platform == 'darwin':
return glob.glob(os.path.expanduser('~/Library/Application Support/Firefox/profiles.ini'))
elif sys.platform.startswith('linux'):
return glob.glob(os.path.expanduser('~/.mozilla/firefox/profiles.ini'))
elif sys.platform == 'win32':
return glob.glob(os.path.join(os.getenv('APPDATA', ''), 'Mozilla/Firefox/profiles.ini'))
else:
raise BrowserCookieError('Unsupported operating system: ' + sys.platform)
def find_cookie_files(self):
profile = self.find_default_profile()
if not profile:
raise BrowserCookieError('Could not find default Firefox profile')
path = self.parse_profile(profile[0])
if not path:
raise BrowserCookieError('Could not find path to default Firefox profile')
cookie_files = glob.glob(os.path.expanduser(path + '/cookies.sqlite'))
if cookie_files:
return cookie_files
else:
raise BrowserCookieError('Failed to find Firefox cookies')
def get_cookies(self):
for cookie_file in self.cookie_files:
with create_local_copy(cookie_file) as tmp_cookie_file:
con = sqlite3.connect(tmp_cookie_file)
cur = con.cursor()
cur.execute('select host, path, isSecure, expiry, name, value from moz_cookies')
for item in cur.fetchall():
yield create_cookie(*item)
con.close()
# current sessions are saved in sessionstore.js/recovery.json/recovery.jsonlz4
session_files = (os.path.join(os.path.dirname(cookie_file), 'sessionstore.js'),
os.path.join(os.path.dirname(cookie_file), 'sessionstore-backups', 'recovery.js'),
os.path.join(os.path.dirname(cookie_file), 'sessionstore-backups', 'recovery.json'),
os.path.join(os.path.dirname(cookie_file), 'sessionstore-backups', 'recovery.jsonlz4'))
for file_path in session_files:
if os.path.exists(file_path):
if file_path.endswith('4'):
try:
session_file = open(file_path, 'rb')
# skip the first 8 bytes to avoid decompress failure (custom Mozilla header)
session_file.seek(8)
json_data = json.loads(lz4.block.decompress(session_file.read()).decode())
except IOError as e:
print('Could not read file:', str(e))
except ValueError as e:
print('Error parsing Firefox session file:', str(e))
else:
try:
json_data = json.loads(open(file_path, 'rb').read().decode('utf-8'))
except IOError as e:
print('Could not read file:', str(e))
except ValueError as e:
print('Error parsing firefox session JSON:', str(e))
if 'json_data' in locals():
expires = str(int(time.time()) + 3600 * 24 * 7)
for window in json_data.get('windows', []):
for cookie in window.get('cookies', []):
yield create_cookie(cookie.get('host', ''), cookie.get('path', ''), False, expires, cookie.get('name', ''), cookie.get('value', ''))
else:
print('Could not find any Firefox session files')
class Safari(BrowserCookieLoader):
def __str__(self):
return 'safari'
def find_cookie_files(self):
if (sys.platform != 'darwin'): # checks if using OSX
BrowserCookieError('Safari is only available on OSX')
else:
cookie_files = glob.glob(os.path.expanduser('~/Library/Cookies')) # no actual use of cookie files because we only have need one specific cookie file
if cookie_files:
return cookie_files
else:
raise BrowserCookieError('Failed to find Safari cookies')
def get_cookies(self):
FilePath = os.path.expanduser('~/Library/Cookies/Cookies.binarycookies')
try:
binary_file = open(FilePath, 'rb')
except IOError:
BrowserCookieError('File Not Found :' + FilePath)
exit()
binary_file.read(4)# will equal 'cook', which stands for cookies
num_pages = unpack('>i', binary_file.read(4))[0]
page_sizes = []
for _ in range(num_pages):
page_sizes.append(unpack('>i', binary_file.read(4))[0])
pages = []
for ps in page_sizes:
pages.append(binary_file.read(ps))
for page in pages:
page = StringIO(page)
page.read(4)
num_cookies = unpack('<i', page.read(4))[0]
cookie_offsets = []
for _ in range(num_cookies):
cookie_offsets.append(unpack('<i', page.read(4))[0])
page.read(4)
cookie = ''
for offset in cookie_offsets:
page.seek(offset)
cookiesize = unpack('<i', page.read(4))[0]
cookie = StringIO(page.read(cookiesize))
cookie.read(4)
flags = unpack('<i', cookie.read(4))[0]
cookie_flags = ''
if flags == 0:
cookie_flags = False # if nothing at all
if flags == 1:
cookie_flags = True # if Secure
elif flags == 4:
cookie_flags = False # if Http only
elif flags == 5:
cookie_flags = True # if Secure and Http only
else:
cookie_flags = False # if Unknown
cookie.read(4)
urloffset = unpack('<i', cookie.read(4))[0]
nameoffset = unpack('<i', cookie.read(4))[0]
pathoffset = unpack('<i', cookie.read(4))[0]
valueoffset = unpack('<i', cookie.read(4))[0]
expiry_date = str(int(unpack('<d', cookie.read(8))[0] + 978307200)) # 978307200 because mac's time starts at: 2001, 1, 1
# create_date = str(int(unpack('<d', cookie.read(8))[0] + 978307200)) no need of it here...
# endofcookie = cookie.read(8) no need it either...
cookie.seek(urloffset - 4)
host = ''
u = cookie.read(1)
while unpack('<b', u)[0] != 0:
host = host + u.decode("utf-8") # in bytes have to be decoded
u = cookie.read(1)
cookie.seek(nameoffset - 4)
name = ''
n = cookie.read(1)
while unpack('<b', n)[0] != 0:
name = name + n.decode("utf-8")
n = cookie.read(1)
cookie.seek(pathoffset - 4)
path = ''
pa = cookie.read(1)
while unpack('<b', pa)[0] != 0:
path = path + pa.decode("utf-8")
pa = cookie.read(1)
cookie.seek(valueoffset - 4)
value = ''
va = cookie.read(1)
while unpack('<b', va)[0] != 0:
value = value + va.decode("utf-8")
va = cookie.read(1)
yield create_cookie(host, path, cookie_flags, expiry_date, name, value)
binary_file.close()
def create_cookie(host, path, secure, expires, name, value):
"""Shortcut function to create a cookie
"""
return cookielib.Cookie(0, name, value, None, False, host, host.startswith('.'), host.startswith('.'), path, True, secure, expires, False, None, None, {})
def chrome(cookie_file=None):
"""Returns a cookiejar of the cookies used by Chrome
"""
return Chrome(cookie_file).load()
def firefox(cookie_file=None):
"""Returns a cookiejar of the cookies and sessions used by Firefox
"""
return Firefox(cookie_file).load()
def safari(cookie_file=None):
"""Returns a cookiejar of the cookies used by safari
"""
return Safari(cookie_file).load()
def _get_cookies():
'''Return all cookies from all browsers'''
for klass in [Chrome, Firefox]:
try:
for cookie in klass().get_cookies():
yield cookie
except BrowserCookieError:
pass
def load():
"""Try to load cookies from all supported browsers and return combined cookiejar
"""
cookie_jar = cookielib.CookieJar()
for cookie in sorted(_get_cookies(), key=lambda cookie: cookie.expires):
cookie_jar.set_cookie(cookie)
return cookie_jar