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
|
import json
from datetime import datetime
NORMAL_MIME = (
'application/pdf',
'application/postscript',
'text/html',
'text/xml',
)
def normalize_mime(raw):
raw = raw.lower()
for norm in NORMAL_MIME:
if raw.startswith(norm):
return norm
# Special cases
if raw.startswith('application/xml'):
return 'text/xml'
if raw.startswith('application/x-pdf'):
return 'application/pdf'
return None
def test_normalize_mime():
assert normalize_mime("asdf") is None
assert normalize_mime("application/pdf") == "application/pdf"
assert normalize_mime("application/pdf+journal") == "application/pdf"
assert normalize_mime("Application/PDF") == "application/pdf"
assert normalize_mime("application/p") is None
assert normalize_mime("application/xml+stuff") == "text/xml"
assert normalize_mime("application/x-pdf") == "application/pdf"
assert normalize_mime("application/x-html") is None
def parse_cdx_line(raw_cdx):
cdx = raw_cdx.split()
if len(cdx) < 11:
return None
surt = cdx[0]
dt = cdx[1]
url = cdx[2]
mime = normalize_mime(cdx[3])
http_status = cdx[4]
key = cdx[5]
c_size = cdx[8]
offset = cdx[9]
warc = cdx[10]
if not (key.isalnum() and c_size.isdigit() and offset.isdigit()
and http_status == "200" and len(key) == 32 and dt.isdigit()
and mime != None):
return None
if '-' in (surt, dt, url, mime, http_status, key, c_size, offset, warc):
return None
key = "sha1:{}".format(key)
info = dict(surt=surt, dt=dt, url=url, c_size=int(c_size),
offset=int(offset), warc=warc)
warc_file = warc.split('/')[-1]
try:
dt_iso = datetime.strptime(dt, "%Y%m%d%H%M%S").isoformat()
except Exception:
return None
# 'i' intentionally not set
heritrix = dict(u=url, d=dt_iso, f=warc_file, o=int(offset), c=1)
return {'key': key, 'file:mime': mime, 'file:cdx': info, 'f:c': heritrix}
def parse_ungrobided_line(raw_line):
line = raw_line.strip().split("\t")
if len(line) != 4:
return None
key = line[0]
mime = normalize_mime(line[2])
try:
f_c = json.loads(line[1])
cdx = json.loads(line[3])
except json.JSONDecodeError:
return None
if not (key[5:].isalnum() and len(key) == 37 and mime != None):
print(mime)
print(key)
print("FAIL")
return None
if '-' in (key, mime, f_c, cdx):
return None
return {'key': key, 'file:mime': mime, 'file:cdx': cdx, 'f:c': f_c}
|