-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathday07.py
executable file
·144 lines (114 loc) · 2.38 KB
/
day07.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
#!/usr/bin/env python3
from utils.all import *
advent.setup(2022, 7)
DEBUG = 'debug' in map(str.lower, sys.argv)
if not DEBUG:
fin = advent.get_input()
else:
fin = io.StringIO('''\
$ cd /
$ ls
dir a
14848514 b.txt
8504156 c.dat
dir d
$ cd a
$ ls
dir e
29116 f
2557 g
62596 h.lst
$ cd e
$ ls
584 i
$ cd ..
$ cd ..
$ cd d
$ ls
4060174 j
8033020 d.log
5626152 d.ext
7214296 k
''')
try: lines = get_lines(fin); fin.seek(0, 0)
except: pass
ans = 0
root = {}
parent = defaultdict(tuple)
t = root
cur = None
lising = False
for line in map(str.strip, lines):
if line.startswith('$'):
listing = False
if line.startswith('$ cd'):
nxt = line[len('$ cd '):]
if nxt == '..':
cur, t = parent[cur]
else:
parent[nxt] = (cur, t)
t[nxt] = {}
cur, t = nxt, t[nxt]
elif line == '$ ls':
listing = True
continue
if listing:
n, name = line.split()
if n.isdigit():
n = int(n)
t[name] = n
elif n == 'dir':
t[name] = {}
else:
assert False, line
# dump_dict(parent)
# eprint('root: --------------------------------')
# dump_dict(root)
# eprint('--------------------------------')
sizes = {}
directories = {'/'}
def dfs(t, name, path):
# eprint('>', name, t, path)
if isinstance(t, int):
return t
directories.add(path)
tot = 0
for subname, subt in t.items():
# eprint('>>', subname, subt)
if name == '/':
p = '/' + subname
else:
p = path + '/' + subname
tot += dfs(subt, subname, p)
sizes[path] = tot
return tot
dfs(root['/'], '/', '/')
# dump_iterable([(v, k) for k, v in sizes.items()])
# eprint('--------------------------------')
# dump_iterable(directories)
for v in sizes.values():
if v <= 100000:
ans += v
# 1001106 wrong
advent.print_answer(1, ans)
tot = sizes['/']
free = 70000000 - tot
need = 30000000 - free
# eprint('tot: ', tot)
# eprint('free:', free)
# eprint('need:', need)
mmin = float('inf')
for name, sz in sizes.items():
if name in directories and sz >= need and sz < mmin:
mmin = sz
# No idea why, but the next smallest size after mmin was correct. Probably some
# error in dfs() or in the input parsing that somehow still got me the right
# value for part 1
second = float('inf')
for name, sz in sizes.items():
if name in directories and sz > mmin and sz < second:
second = sz
# dump_iterable(sorted(sizes.items(), key=lambda kv: kv[1]))
# 3487907 wrong
# 3866390 correct for whatever reason...
advent.print_answer(2, second)