dl_cleanup.py 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  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. ".orig.tar.gz",
  85. ".orig.tar.bz2",
  86. ".orig.tar.xz",
  87. ".zip",
  88. ".tgz",
  89. ".tbz",
  90. ".txz",
  91. )
  92. versionRegex = (
  93. (re.compile(r"(gcc[-_]\d+)\.(\d+)\.(\d+)"), parseVer_12), # gcc.1.2
  94. (re.compile(r"(linux[-_]\d+\.\d+)\.(\d+)"), parseVer_r), # linux.1
  95. (re.compile(r"(.+)[-_](\d+)\.(\d+)\.(\d+)\.(\d+)"), parseVer_1234), # xxx-1.2.3.4
  96. (
  97. re.compile(r"(.+)[-_](\d\d\d\d)-?(\d\d)-?(\d\d)-"),
  98. parseVer_ymd_GIT_SHASUM,
  99. ), # xxx-YYYY-MM-DD-GIT_SHASUM
  100. (re.compile(r"(.+)[-_](\d\d\d\d)-?(\d\d)-?(\d\d)"), parseVer_ymd), # xxx-YYYY-MM-DD
  101. (re.compile(r"(.+)[-_]([0-9a-fA-F]{40,40})"), parseVer_GIT), # xxx-GIT_SHASUM
  102. (re.compile(r"(.+)[-_](\d+)\.(\d+)\.(\d+)(\w?)"), parseVer_123), # xxx-1.2.3a
  103. (re.compile(r"(.+)[-_]v(\d+)\.(\d+)\.(\d+)(\w?)"), parseVer_123), # xxx-v1.2.3a
  104. (re.compile(r"(.+)[-_](\d+)_(\d+)_(\d+)"), parseVer_123), # xxx-1_2_3
  105. (re.compile(r"(.+)[-_](\d+)\.(\d+)(\w?)"), parseVer_12), # xxx-1.2a
  106. (re.compile(r"(.+)[-_]v(\d+)\.(\d+)(\w?)"), parseVer_12), # xxx-v1.2a
  107. (re.compile(r"(.+)[-_]r?(\d+)"), parseVer_r), # xxx-r1111
  108. )
  109. blacklist = [
  110. ("wl_apsta", re.compile(r"wl_apsta.*")),
  111. (".fw", re.compile(r".*\.fw")),
  112. (".arm", re.compile(r".*\.arm")),
  113. (".bin", re.compile(r".*\.bin")),
  114. ("rt-firmware", re.compile(r"RT[\d\w]+_Firmware.*")),
  115. ]
  116. class EntryParseError(Exception):
  117. pass
  118. class Entry:
  119. def __init__(self, directory, builddir, filename):
  120. self.directory = directory
  121. self.filename = filename
  122. self.builddir = builddir
  123. self.progname = ""
  124. self.fileext = ""
  125. self.filenoext = ""
  126. for ext in extensions:
  127. if filename.endswith(ext):
  128. filename = filename[0 : 0 - len(ext)]
  129. self.filenoext = filename
  130. self.fileext = ext
  131. break
  132. else:
  133. print(self.filename, "has an unknown file-extension")
  134. raise EntryParseError("ext")
  135. for (regex, parseVersion) in versionRegex:
  136. match = regex.match(filename)
  137. if match:
  138. (self.progname, self.version) = parseVersion(
  139. match, directory + "/" + filename + self.fileext
  140. )
  141. break
  142. else:
  143. print(self.filename, "has an unknown version pattern")
  144. raise EntryParseError("ver")
  145. def getPath(self):
  146. return (self.directory + "/" + self.filename).replace("//", "/")
  147. def getBuildPaths(self):
  148. paths = []
  149. for subdir in os.scandir(self.builddir):
  150. package_build_dir = os.path.join(subdir.path, self.filenoext)
  151. if os.path.exists(package_build_dir):
  152. paths.append(package_build_dir)
  153. return paths
  154. def deleteFile(self):
  155. path = self.getPath()
  156. print("Deleting", path)
  157. if not opt_dryrun:
  158. os.unlink(path)
  159. def deleteBuildDir(self):
  160. paths = self.getBuildPaths()
  161. for path in paths:
  162. print("Deleting BuildDir", path)
  163. if not opt_dryrun:
  164. shutil.rmtree(path)
  165. def __ge__(self, y):
  166. return self.version >= y.version
  167. def usage():
  168. print("OpenWrt download directory cleanup utility")
  169. print("Usage: " + sys.argv[0] + " [OPTIONS] <path/to/dl>")
  170. print("")
  171. print(" -d|--dry-run Do a dry-run. Don't delete any files")
  172. print(" -B|--show-blacklist Show the blacklist and exit")
  173. print(" -w|--whitelist ITEM Remove ITEM from blacklist")
  174. print(
  175. " -D|--download-dir Provide path to dl dir to clean also the build directory"
  176. )
  177. print(
  178. " -b|--build-dir Provide path to build dir to clean also the build directory"
  179. )
  180. def main(argv):
  181. global opt_dryrun
  182. try:
  183. (opts, args) = getopt.getopt(
  184. argv[1:],
  185. "hdBwDb:",
  186. [
  187. "help",
  188. "dry-run",
  189. "show-blacklist",
  190. "whitelist=",
  191. "download-dir=",
  192. "build-dir=",
  193. ],
  194. )
  195. except getopt.GetoptError as e:
  196. usage()
  197. return 1
  198. directory = "dl/"
  199. builddir = "build_dir/"
  200. for (o, v) in opts:
  201. if o in ("-h", "--help"):
  202. usage()
  203. return 0
  204. if o in ("-d", "--dry-run"):
  205. opt_dryrun = True
  206. if o in ("-w", "--whitelist"):
  207. for i in range(0, len(blacklist)):
  208. (name, regex) = blacklist[i]
  209. if name == v:
  210. del blacklist[i]
  211. break
  212. else:
  213. print("Whitelist error: Item", v, "is not in blacklist")
  214. return 1
  215. if o in ("-B", "--show-blacklist"):
  216. for (name, regex) in blacklist:
  217. sep = "\t\t"
  218. if len(name) >= 8:
  219. sep = "\t"
  220. print("%s%s(%s)" % (name, sep, regex.pattern))
  221. return 0
  222. if o in ("-D", "--download-dir"):
  223. directory = v
  224. if o in ("-b", "--build-dir"):
  225. builddir = v
  226. if not os.path.exists(directory):
  227. print("Can't find dl path", directory)
  228. return 1
  229. if not os.path.exists(builddir):
  230. print("Can't find dl path", builddir)
  231. return 1
  232. # Create a directory listing and parse the file names.
  233. entries = []
  234. for filename in os.listdir(directory):
  235. if filename == "." or filename == "..":
  236. continue
  237. for (name, regex) in blacklist:
  238. if regex.match(filename):
  239. if opt_dryrun:
  240. print(filename, "is blacklisted")
  241. break
  242. else:
  243. try:
  244. entries.append(Entry(directory, builddir, filename))
  245. except EntryParseError as e:
  246. pass
  247. # Create a map of programs
  248. progmap = {}
  249. for entry in entries:
  250. if entry.progname in progmap.keys():
  251. progmap[entry.progname].append(entry)
  252. else:
  253. progmap[entry.progname] = [
  254. entry,
  255. ]
  256. # Traverse the program map and delete everything but the last version
  257. for prog in progmap:
  258. lastVersion = None
  259. versions = progmap[prog]
  260. for version in versions:
  261. if lastVersion is None or version >= lastVersion:
  262. lastVersion = version
  263. if lastVersion:
  264. for version in versions:
  265. if version is not lastVersion:
  266. version.deleteFile()
  267. if builddir:
  268. version.deleteBuildDir()
  269. if opt_dryrun:
  270. print("Keeping", lastVersion.getPath())
  271. return 0
  272. if __name__ == "__main__":
  273. sys.exit(main(sys.argv))