dl_cleanup.py 9.4 KB

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