-
Notifications
You must be signed in to change notification settings - Fork 0
/
partition_packages_by_abandoned.py
executable file
·108 lines (80 loc) · 3.59 KB
/
partition_packages_by_abandoned.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
#!/usr/bin/env python3
# 1st-party
import calendar
import datetime
import json
import logging
import os
import sys
# 2nd-party
import package_cache
OUTPUT_DIR = '/var/experiments-output/'
# Data source 3: A map of a project to the date (not time) of when it last
# added, updated or removed a package.
PACKAGE_LAST_MODIFIED_FILENAME = os.path.join(OUTPUT_DIR, 'package_cache.json')
# The experiment is only valid since the following Unix timestamp.
SINCE_TIMESTAMP = 1395360000
SINCE_DATETIME = datetime.datetime.utcfromtimestamp(SINCE_TIMESTAMP)
def assert_datetime_has_no_time(some_datetime):
assert some_datetime.hour == 0
assert some_datetime.minute == 0
assert some_datetime.second == 0
assert some_datetime.microsecond == 0
assert some_datetime.tzinfo is None
def get_timestamp_before_compromise(time_delta):
assert_datetime_has_no_time(SINCE_DATETIME)
before_datetime = SINCE_DATETIME-time_delta
assert_datetime_has_no_time(before_datetime)
assert before_datetime <= SINCE_DATETIME
# NOTE: necessary to get GMT timestamp from a "naive" datetime object
return calendar.timegm(before_datetime.utctimetuple())
def partition(abandon_earlier_than_timedelta):
safe_packages, unsafe_packages = set(), set()
abandon_earlier_than_timestamp = \
get_timestamp_before_compromise(abandon_earlier_than_timedelta)
assert abandon_earlier_than_timestamp < SINCE_TIMESTAMP
abandon_earlier_than_datetime = \
datetime.datetime.utcfromtimestamp(abandon_earlier_than_timestamp)
logging.info('Projects that updated before {} '\
'({} ago) are considered abandoned'\
.format(abandon_earlier_than_datetime,
abandon_earlier_than_timedelta))
with open(PACKAGE_LAST_MODIFIED_FILENAME, 'rt') as fp:
packages_list = json.load(fp)
for package in packages_list:
# Get timestamps of when the project added/updated/removed a package.
package_timestamps = packages_list[package]
# Get timestamp, if ANY, of when the project LAST added/updated/removed a
# package BEFORE the compromise.
package_timestamp = \
package_cache.get_last_timestamp_before_compromise(package_timestamps,
SINCE_TIMESTAMP)
if package_timestamp:
if package_timestamp < abandon_earlier_than_timestamp:
safe_packages.add(package)
else:
unsafe_packages.add(package)
else:
unsafe_packages.add(package)
assert len(safe_packages & unsafe_packages) == 0
logging.info('Safe abandoned projects: {:,}'.format(len(safe_packages)))
logging.info('Vuln abandoned projects: {:,}'.format(len(unsafe_packages)))
return safe_packages, unsafe_packages
if __name__ == '__main__':
# rw for owner and group but not others
os.umask(0o07)
# What is the number of days with which to partition packages?
assert len(sys.argv) == 2
number_of_days = int(sys.argv[1])
assert number_of_days > 0
time_delta = datetime.timedelta(days=int(sys.argv[1]))
safe_packages, unsafe_packages = partition(time_delta)
safe_packages = sorted(list(safe_packages))
unsafe_packages = sorted(list(unsafe_packages))
output_json_filename = 'partition_packages_by_abandoned.{}days.json'\
.format(time_delta.days)
output_json_filename = os.path.join(OUTPUT_DIR, output_json_filename)
output_json = {'safe': safe_packages, 'unsafe': unsafe_packages}
with open(output_json_filename, 'wt') as output_json_file:
json.dump(output_json, output_json_file, sort_keys=True, indent=4,
separators=(',', ': '))