forked from niv/neverwinter.nim
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnwn_nwsync_prune.nim
153 lines (114 loc) · 5.09 KB
/
nwn_nwsync_prune.nim
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
import shared
let ARGS = DOC """
This utility will perform housekeeping on a nwsync repository.
It will:
- Make sure `latest` is a valid pointer, if present.
- Warns about manifests missing metadata.
- Prune all data files not contained in any stored manifests.
- Warn about missing data files.
- Clean up the directory structure.
Usage:
$0 [options] <root>
$USAGE
Options:
-n --dry-run Simulate, don't actually do anything.
--min-age AGE Do not prune data files that have been written within
AGE seconds. 2 weeks = [default: 1209600]
$OPT
"""
import std/[os, strutils, logging, critbits, sequtils, sets, times]
import neverwinter/nwsync
import neverwinter/nwsync/private/libshared
proc isSha1*(candidate: string): bool =
candidate.len == 40 and candidate.count({'a'..'f', '0'..'9'}) == 40
addHandler newConsoleLogger(fmtStr = verboseFmtStr)
setLogFilter(if ARGS["--verbose"]: lvlDebug elif ARGS["--quiet"]: lvlWarn else: lvlInfo)
proc act*(hr: varargs[string, `$`], runnable: proc()) =
if not ARGS["--dry-run"]: runnable()
else: notice "Dry run: ", foldl(@hr, a & " " & b)
let root = $ARGS["<root>"]
let minAge: int64 = parseBiggestInt($ARGS["--min-age"]).clamp(0, int64.high)
### CHECK: Ensure `latest` points to a valid manifest file
proc ensureLatestHeadIsValid*(rootDirectory: string) =
info "Ensuring `latest` is valid:"
if fileExists(rootDirectory / "latest"):
let latest = readFile(rootDirectory / "latest").strip
if not fileExists(rootDirectory / "manifests" / latest):
error "`latest` points to non-existing manifest ", latest
else:
info "OK: ", latest
else:
info "`latest` does not exist (advisory)"
ensureLatestHeadIsValid(root)
### CHECK: look for missing data files
### MAINTENANCE: Remove all unreferenced files
proc updateManifestMetaData*(rootDirectory: string, mfh: string, mf: Manifest) =
if not fileExists(rootDirectory / "manifests" / mfh & ".json"):
error "Manifest ", mfh, " found, but has no accompanying metadata file. ",
"You need to fix this manually by re-generating the manifest from source."
proc pruneUnreferencedFiles*(rootDirectory: string) =
info "Checking for orphaned and missing files"
# Manifests we have in storage: hash => Manifest
info "Loading all manifests (this may take a good while)"
var manifestsInRepository: CritBitTree[Manifest]
for pa in walkDir(rootDirectory / "manifests"):
let mfh = pa.path.extractFilename
if not isSha1(mfh): continue
info "Reading manifest ", mfh
let mf = readManifest(pa.path)
updateManifestMetaData(rootDirectory, mfh, mf)
manifestsInRepository[mfh] = mf
info "Manifest ", mfh,
" containing ", formatSize(int mf.totalSize()),
" (deduplicated to ", formatSize(int mf.deduplicatedSize()),
") in ",
mf.entries.len, " files"
# Referenced files: fileHash => [manifestHashes..]
var referenced: CritBitTree[CritBitTree[void]]
for mfh, mf in manifestsInRepository:
for mfRes, mfEntry in mf.entries:
let entryHash = toLowerAscii($mfEntry.sha1)
if not referenced.hasKey(entryHash):
var newTree: CritBitTree[void]
referenced[entryHash] = newTree
referenced[entryHash].incl(mfh)
let referencedHashes = toHashSet toSeq referenced.keys
info "Found ", referenced.len, " referenced files in all manifests"
let inStorage = getFilesInStorage(rootDirectory)
let inStorageHashes = toHashSet toSeq inStorage.keys
info "Found ", inStorage.len, " files in storage"
let orphans = inStorageHashes - referencedHashes
let missing = referencedHashes - inStorageHashes
info "Orphans (not referenced, but in storage): ", orphans.len
info "Missing (referenced, but not in storage): ", missing.len
for m in missing:
let mfh = toSeq(referenced[m].items).
#mapIt("($1, $2)".format(it, manifestsInRepository[it].entries[m].resref)).
join(", ")
error "MISSING: ", m, ", referenced in ", mfh
let now = getTime()
for o in orphans:
let f = inStorage[o]
let age = now - getLastModificationTime(f)
if age.inSEconds > minAge:
act("Would delete file: ", f) do (): removeFile(f)
pruneUnreferencedFiles(root)
### MAINTENANCE: remove all empty hash dirs
proc pruneEmptyDirectories*(dir: string) =
for pc in walkDir(dir, relative = true):
if pc.kind == pcDir and pc.path.len == 2: #todo: unsuck
# Recurse first.
pruneEmptyDirectories(dir / pc.path)
let contents = toSeq(walkDir(dir / pc.path))
if contents.len == 0:
act("Would remove directory: ", dir / pc.path) do(): removeDir(dir / pc.path)
pruneEmptyDirectories(root / "data" / "sha1")
### MAINTENANCE: remove all outdated metadata files
proc pruneDanglingMetadata*(rootDirectory: string) =
for pa in walkDir(rootDirectory / "manifests"):
let sp = pa.path.splitFile
if not isSha1(sp.name): continue
if sp.ext == ".json" and not fileExists(rootDirectory / "manifests" / sp.name):
notice "Removing dangling manifest metadata with no manifest: ", sp.name
act("Would delete file: ", pa.path) do (): removeFile(pa.path)
pruneDanglingMetadata(root)