ncurses: Fix path in ncursesw.pc
[librecmc/librecmc.git] / scripts / dl_cleanup.py
1 #!/usr/bin/env python3
2 """
3 # libreCMC download directory cleanup utility.
4 # Delete all but the very last version of the program tarballs.
5 #
6 # Copyright (C) 2010-2015 Michael Buesch <m@bues.ch>
7 # Copyright (C) 2013-2015 OpenWrt.org
8 """
9
10 from __future__ import print_function
11
12 import sys
13 import os
14 import re
15 import getopt
16 import shutil
17
18 # Commandline options
19 opt_dryrun = False
20
21
22 def parseVer_1234(match, filepath):
23     progname = match.group(1)
24     progversion = (
25         (int(match.group(2)) << 64)
26         | (int(match.group(3)) << 48)
27         | (int(match.group(4)) << 32)
28         | (int(match.group(5)) << 16)
29     )
30     return (progname, progversion)
31
32
33 def parseVer_123(match, filepath):
34     progname = match.group(1)
35     try:
36         patchlevel = match.group(5)
37     except IndexError as e:
38         patchlevel = None
39     if patchlevel:
40         patchlevel = ord(patchlevel[0])
41     else:
42         patchlevel = 0
43     progversion = (
44         (int(match.group(2)) << 64)
45         | (int(match.group(3)) << 48)
46         | (int(match.group(4)) << 32)
47         | patchlevel
48     )
49     return (progname, progversion)
50
51
52 def parseVer_12(match, filepath):
53     progname = match.group(1)
54     try:
55         patchlevel = match.group(4)
56     except IndexError as e:
57         patchlevel = None
58     if patchlevel:
59         patchlevel = ord(patchlevel[0])
60     else:
61         patchlevel = 0
62     progversion = (int(match.group(2)) << 64) | (int(match.group(3)) << 48) | patchlevel
63     return (progname, progversion)
64
65
66 def parseVer_r(match, filepath):
67     progname = match.group(1)
68     progversion = int(match.group(2)) << 64
69     return (progname, progversion)
70
71
72 def parseVer_ymd_GIT_SHASUM(match, filepath):
73     progname = match.group(1)
74     progversion = (
75         (int(match.group(2)) << 64)
76         | (int(match.group(3)) << 48)
77         | (int(match.group(4)) << 32)
78     )
79     return (progname, progversion)
80
81
82 def parseVer_ymd(match, filepath):
83     progname = match.group(1)
84     progversion = (
85         (int(match.group(2)) << 64)
86         | (int(match.group(3)) << 48)
87         | (int(match.group(4)) << 32)
88     )
89     return (progname, progversion)
90
91
92 def parseVer_GIT(match, filepath):
93     progname = match.group(1)
94     st = os.stat(filepath)
95     progversion = int(st.st_mtime) << 64
96     return (progname, progversion)
97
98
99 extensions = (
100     ".tar.gz",
101     ".tar.bz2",
102     ".tar.xz",
103     ".orig.tar.gz",
104     ".orig.tar.bz2",
105     ".orig.tar.xz",
106     ".zip",
107     ".tgz",
108     ".tbz",
109     ".txz",
110 )
111
112 versionRegex = (
113     (re.compile(r"(gcc[-_]\d+)\.(\d+)\.(\d+)"), parseVer_12),  # gcc.1.2
114     (re.compile(r"(linux[-_]\d+\.\d+)\.(\d+)"), parseVer_r),  # linux.1
115     (re.compile(r"(.+)[-_](\d+)\.(\d+)\.(\d+)\.(\d+)"), parseVer_1234),  # xxx-1.2.3.4
116     (
117         re.compile(r"(.+)[-_](\d\d\d\d)-?(\d\d)-?(\d\d)-"),
118         parseVer_ymd_GIT_SHASUM,
119     ),  # xxx-YYYY-MM-DD-GIT_SHASUM
120     (re.compile(r"(.+)[-_](\d\d\d\d)-?(\d\d)-?(\d\d)"), parseVer_ymd),  # xxx-YYYY-MM-DD
121     (re.compile(r"(.+)[-_]([0-9a-fA-F]{40,40})"), parseVer_GIT),  # xxx-GIT_SHASUM
122     (re.compile(r"(.+)[-_](\d+)\.(\d+)\.(\d+)(\w?)"), parseVer_123),  # xxx-1.2.3a
123     (re.compile(r"(.+)[-_]v(\d+)\.(\d+)\.(\d+)(\w?)"), parseVer_123),  # xxx-v1.2.3a
124     (re.compile(r"(.+)[-_](\d+)_(\d+)_(\d+)"), parseVer_123),  # xxx-1_2_3
125     (re.compile(r"(.+)[-_](\d+)\.(\d+)(\w?)"), parseVer_12),  # xxx-1.2a
126     (re.compile(r"(.+)[-_]v(\d+)\.(\d+)(\w?)"), parseVer_12),  # xxx-v1.2a
127     (re.compile(r"(.+)[-_]r?(\d+)"), parseVer_r),  # xxx-r1111
128 )
129
130 blacklist = [
131     ("wl_apsta", re.compile(r"wl_apsta.*")),
132     (".fw", re.compile(r".*\.fw")),
133     (".arm", re.compile(r".*\.arm")),
134     (".bin", re.compile(r".*\.bin")),
135     ("rt-firmware", re.compile(r"RT[\d\w]+_Firmware.*")),
136 ]
137
138
139 class EntryParseError(Exception):
140     pass
141
142
143 class Entry:
144     def __init__(self, directory, builddir, filename):
145         self.directory = directory
146         self.filename = filename
147         self.builddir = builddir
148         self.progname = ""
149         self.fileext = ""
150         self.filenoext = ""
151
152         if os.path.isdir(self.getPath()):
153             self.filenoext = filename
154         else:
155             for ext in extensions:
156                 if filename.endswith(ext):
157                     filename = filename[0 : 0 - len(ext)]
158                     self.filenoext = filename
159                     self.fileext = ext
160                     break
161             else:
162                 print(self.filename, "has an unknown file-extension")
163                 raise EntryParseError("ext")
164         for (regex, parseVersion) in versionRegex:
165             match = regex.match(filename)
166             if match:
167                 (self.progname, self.version) = parseVersion(
168                     match, directory + "/" + filename + self.fileext
169                 )
170                 break
171         else:
172             print(self.filename, "has an unknown version pattern")
173             raise EntryParseError("ver")
174
175     def getPath(self):
176         return (self.directory + "/" + self.filename).replace("//", "/")
177
178     def getBuildPaths(self):
179         paths = []
180         for subdir in os.scandir(self.builddir):
181             package_build_dir = os.path.join(subdir.path, self.filenoext)
182             if os.path.exists(package_build_dir):
183                 paths.append(package_build_dir)
184         return paths
185
186     def deleteFile(self):
187         path = self.getPath()
188         print("Deleting", path)
189         if not opt_dryrun:
190             if os.path.isdir(path):
191                 shutil.rmtree(path)
192             else:
193                 os.unlink(path)
194
195     def deleteBuildDir(self):
196         paths = self.getBuildPaths()
197         for path in paths:
198             print("Deleting BuildDir", path)
199             if not opt_dryrun:
200                     shutil.rmtree(path)
201
202     def __ge__(self, y):
203         return self.version >= y.version
204
205
206 def usage():
207     print("libreCMC download directory cleanup utility")
208     print("Usage: " + sys.argv[0] + " [OPTIONS] <path/to/dl>")
209     print("")
210     print(" -d|--dry-run            Do a dry-run. Don't delete any files")
211     print(" -B|--show-blacklist     Show the blacklist and exit")
212     print(" -w|--whitelist ITEM     Remove ITEM from blacklist")
213     print(
214         " -D|--download-dir       Provide path to dl dir to clean also the build directory"
215     )
216     print(
217         " -b|--build-dir          Provide path to build dir to clean also the build directory"
218     )
219
220
221 def main(argv):
222     global opt_dryrun
223
224     try:
225         (opts, args) = getopt.getopt(
226             argv[1:],
227             "hdBw:D:b:",
228             [
229                 "help",
230                 "dry-run",
231                 "show-blacklist",
232                 "whitelist=",
233                 "download-dir=",
234                 "build-dir=",
235             ],
236         )
237     except getopt.GetoptError as e:
238         usage()
239         return 1
240
241     directory = "dl/"
242     builddir = "build_dir/"
243
244     for (o, v) in opts:
245         if o in ("-h", "--help"):
246             usage()
247             return 0
248         if o in ("-d", "--dry-run"):
249             opt_dryrun = True
250         if o in ("-w", "--whitelist"):
251             for i in range(0, len(blacklist)):
252                 (name, regex) = blacklist[i]
253                 if name == v:
254                     del blacklist[i]
255                     break
256             else:
257                 print("Whitelist error: Item", v, "is not in blacklist")
258                 return 1
259         if o in ("-B", "--show-blacklist"):
260             for (name, regex) in blacklist:
261                 sep = "\t\t"
262                 if len(name) >= 8:
263                     sep = "\t"
264                 print("%s%s(%s)" % (name, sep, regex.pattern))
265             return 0
266         if o in ("-D", "--download-dir"):
267             directory = v
268         if o in ("-b", "--build-dir"):
269             builddir = v
270
271     if args:
272         directory = args[0]
273
274     if not os.path.exists(directory):
275         print("Can't find download directory", directory)
276         return 1
277
278     if not os.path.exists(builddir):
279         print("Can't find build directory", builddir)
280         return 1
281
282     # Create a directory listing and parse the file names.
283     entries = []
284     for filename in os.listdir(directory):
285         if filename == "." or filename == "..":
286             continue
287         for (name, regex) in blacklist:
288             if regex.match(filename):
289                 if opt_dryrun:
290                     print(filename, "is blacklisted")
291                 break
292         else:
293             try:
294                 entries.append(Entry(directory, builddir, filename))
295             except EntryParseError as e:
296                 pass
297
298     # Create a map of programs
299     progmap = {}
300     for entry in entries:
301         if entry.progname in progmap.keys():
302             progmap[entry.progname].append(entry)
303         else:
304             progmap[entry.progname] = [
305                 entry,
306             ]
307
308     # Traverse the program map and delete everything but the last version
309     for prog in progmap:
310         lastVersion = None
311         versions = progmap[prog]
312         for version in versions:
313             if lastVersion:
314                 if os.path.isdir(lastVersion.getPath()) and not os.path.isdir(version.getPath()):
315                     continue
316             if lastVersion is None or version >= lastVersion:
317                 lastVersion = version
318         if lastVersion:
319             for version in versions:
320                 if version is not lastVersion:
321                     version.deleteFile()
322                     if builddir:
323                         version.deleteBuildDir()
324             if opt_dryrun:
325                 print("Keeping", lastVersion.getPath())
326
327     return 0
328
329
330 if __name__ == "__main__":
331     sys.exit(main(sys.argv))