UiWebsocketPlugin.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  1. import re
  2. import time
  3. import cgi
  4. import gevent
  5. from Plugin import PluginManager
  6. from Config import config
  7. from util import helper
  8. from Translate import Translate
  9. if "_" not in locals():
  10. _ = Translate("plugins/OptionalManager/languages/")
  11. bigfile_sha512_cache = {}
  12. @PluginManager.registerTo("UiWebsocket")
  13. class UiWebsocketPlugin(object):
  14. def __init__(self, *args, **kwargs):
  15. self.time_peer_numbers_updated = 0
  16. super(UiWebsocketPlugin, self).__init__(*args, **kwargs)
  17. def actionSiteSign(self, to, privatekey=None, inner_path="content.json", *args, **kwargs):
  18. # Add file to content.db and set it as pinned
  19. content_db = self.site.content_manager.contents.db
  20. content_inner_dir = helper.getDirname(inner_path)
  21. content_db.my_optional_files[self.site.address + "/" + content_inner_dir] = time.time()
  22. if len(content_db.my_optional_files) > 50: # Keep only last 50
  23. oldest_key = min(
  24. content_db.my_optional_files.iterkeys(),
  25. key=(lambda key: content_db.my_optional_files[key])
  26. )
  27. del content_db.my_optional_files[oldest_key]
  28. return super(UiWebsocketPlugin, self).actionSiteSign(to, privatekey, inner_path, *args, **kwargs)
  29. def updatePeerNumbers(self):
  30. self.site.updateHashfield()
  31. content_db = self.site.content_manager.contents.db
  32. content_db.updatePeerNumbers()
  33. self.site.updateWebsocket(peernumber_updated=True)
  34. def addBigfileInfo(self, row):
  35. global bigfile_sha512_cache
  36. content_db = self.site.content_manager.contents.db
  37. site = content_db.sites[row["address"]]
  38. if not site.settings.get("has_bigfile"):
  39. return False
  40. file_key = row["address"] + "/" + row["inner_path"]
  41. sha512 = bigfile_sha512_cache.get(file_key)
  42. file_info = None
  43. if not sha512:
  44. file_info = site.content_manager.getFileInfo(row["inner_path"])
  45. if not file_info or not file_info.get("piece_size"):
  46. return False
  47. sha512 = file_info["sha512"]
  48. bigfile_sha512_cache[file_key] = sha512
  49. if sha512 in site.storage.piecefields:
  50. piecefield = site.storage.piecefields[sha512].tostring()
  51. else:
  52. piecefield = None
  53. if piecefield:
  54. row["pieces"] = len(piecefield)
  55. row["pieces_downloaded"] = piecefield.count("1")
  56. row["downloaded_percent"] = 100 * row["pieces_downloaded"] / row["pieces"]
  57. if row["pieces_downloaded"]:
  58. if not file_info:
  59. file_info = site.content_manager.getFileInfo(row["inner_path"])
  60. row["bytes_downloaded"] = row["pieces_downloaded"] * file_info.get("piece_size", 0)
  61. else:
  62. row["bytes_downloaded"] = 0
  63. row["is_downloading"] = bool(next((inner_path for inner_path in site.bad_files if inner_path.startswith(row["inner_path"])), False))
  64. # Add leech / seed stats
  65. row["peer_seed"] = 0
  66. row["peer_leech"] = 0
  67. for peer in site.peers.itervalues():
  68. if not peer.time_piecefields_updated or sha512 not in peer.piecefields:
  69. continue
  70. peer_piecefield = peer.piecefields[sha512].tostring()
  71. if not peer_piecefield:
  72. continue
  73. if peer_piecefield == "1" * len(peer_piecefield):
  74. row["peer_seed"] += 1
  75. else:
  76. row["peer_leech"] += 1
  77. # Add myself
  78. if piecefield:
  79. if row["pieces_downloaded"] == row["pieces"]:
  80. row["peer_seed"] += 1
  81. else:
  82. row["peer_leech"] += 1
  83. return True
  84. # Optional file functions
  85. def actionOptionalFileList(self, to, address=None, orderby="time_downloaded DESC", limit=10, filter="downloaded"):
  86. if not address:
  87. address = self.site.address
  88. # Update peer numbers if necessary
  89. content_db = self.site.content_manager.contents.db
  90. if time.time() - content_db.time_peer_numbers_updated > 60 * 1 and time.time() - self.time_peer_numbers_updated > 60 * 5:
  91. # Start in new thread to avoid blocking
  92. self.time_peer_numbers_updated = time.time()
  93. gevent.spawn(self.updatePeerNumbers)
  94. if address == "all" and "ADMIN" not in self.permissions:
  95. return self.response(to, {"error": "Forbidden"})
  96. if not self.hasSitePermission(address):
  97. return self.response(to, {"error": "Forbidden"})
  98. if not all([re.match("^[a-z_*/+-]+( DESC| ASC|)$", part.strip()) for part in orderby.split(",")]):
  99. return self.response(to, "Invalid order_by")
  100. if type(limit) != int:
  101. return self.response(to, "Invalid limit")
  102. back = []
  103. content_db = self.site.content_manager.contents.db
  104. wheres = {}
  105. wheres_raw = []
  106. if "bigfile" in filter:
  107. wheres["size >"] = 1024 * 1024 * 10
  108. if "not_downloaded" in filter:
  109. wheres["is_downloaded"] = 0
  110. elif "downloaded" in filter:
  111. wheres_raw.append("(is_downloaded = 1 OR is_pinned = 1)")
  112. if "pinned" in filter:
  113. wheres["is_pinned"] = 1
  114. if address == "all":
  115. join = "LEFT JOIN site USING (site_id)"
  116. else:
  117. wheres["site_id"] = content_db.site_ids[address]
  118. join = ""
  119. if wheres_raw:
  120. query_wheres_raw = "AND" + " AND ".join(wheres_raw)
  121. else:
  122. query_wheres_raw = ""
  123. query = "SELECT * FROM file_optional %s WHERE ? %s ORDER BY %s LIMIT %s" % (join, query_wheres_raw, orderby, limit)
  124. for row in content_db.execute(query, wheres):
  125. row = dict(row)
  126. if address != "all":
  127. row["address"] = address
  128. if row["size"] > 1024 * 1024:
  129. has_info = self.addBigfileInfo(row)
  130. else:
  131. has_info = False
  132. if not has_info:
  133. if row["is_downloaded"]:
  134. row["bytes_downloaded"] = row["size"]
  135. row["downloaded_percent"] = 100
  136. else:
  137. row["bytes_downloaded"] = 0
  138. row["downloaded_percent"] = 0
  139. back.append(row)
  140. self.response(to, back)
  141. def actionOptionalFileInfo(self, to, inner_path):
  142. content_db = self.site.content_manager.contents.db
  143. site_id = content_db.site_ids[self.site.address]
  144. # Update peer numbers if necessary
  145. if time.time() - content_db.time_peer_numbers_updated > 60 * 1 and time.time() - self.time_peer_numbers_updated > 60 * 5:
  146. # Start in new thread to avoid blocking
  147. self.time_peer_numbers_updated = time.time()
  148. gevent.spawn(self.updatePeerNumbers)
  149. query = "SELECT * FROM file_optional WHERE site_id = :site_id AND inner_path = :inner_path LIMIT 1"
  150. res = content_db.execute(query, {"site_id": site_id, "inner_path": inner_path})
  151. row = next(res, None)
  152. if row:
  153. row = dict(row)
  154. if row["size"] > 1024 * 1024:
  155. row["address"] = self.site.address
  156. self.addBigfileInfo(row)
  157. self.response(to, row)
  158. else:
  159. self.response(to, None)
  160. def setPin(self, inner_path, is_pinned, address=None):
  161. if not address:
  162. address = self.site.address
  163. if not self.hasSitePermission(address):
  164. return {"error": "Forbidden"}
  165. site = self.server.sites[address]
  166. site.content_manager.setPin(inner_path, is_pinned)
  167. return "ok"
  168. def actionOptionalFilePin(self, to, inner_path, address=None):
  169. if type(inner_path) is not list:
  170. inner_path = [inner_path]
  171. back = self.setPin(inner_path, 1, address)
  172. num_file = len(inner_path)
  173. if back == "ok":
  174. if num_file == 1:
  175. self.cmd("notification", ["done", _["Pinned %s"] % cgi.escape(helper.getFilename(inner_path[0])), 5000])
  176. else:
  177. self.cmd("notification", ["done", _["Pinned %s files"] % num_file, 5000])
  178. self.response(to, back)
  179. def actionOptionalFileUnpin(self, to, inner_path, address=None):
  180. if type(inner_path) is not list:
  181. inner_path = [inner_path]
  182. back = self.setPin(inner_path, 0, address)
  183. num_file = len(inner_path)
  184. if back == "ok":
  185. if num_file == 1:
  186. self.cmd("notification", ["done", _["Removed pin from %s"] % cgi.escape(helper.getFilename(inner_path[0])), 5000])
  187. else:
  188. self.cmd("notification", ["done", _["Removed pin from %s files"] % num_file, 5000])
  189. self.response(to, back)
  190. def actionOptionalFileDelete(self, to, inner_path, address=None):
  191. if not address:
  192. address = self.site.address
  193. if not self.hasSitePermission(address):
  194. return self.response(to, {"error": "Forbidden"})
  195. site = self.server.sites[address]
  196. content_db = site.content_manager.contents.db
  197. site_id = content_db.site_ids[site.address]
  198. res = content_db.execute("SELECT * FROM file_optional WHERE ? LIMIT 1", {"site_id": site_id, "inner_path": inner_path, "is_downloaded": 1})
  199. row = next(res, None)
  200. if not row:
  201. return self.response(to, {"error": "Not found in content.db"})
  202. removed = site.content_manager.optionalRemoved(inner_path, row["hash_id"], row["size"])
  203. # if not removed:
  204. # return self.response(to, {"error": "Not found in hash_id: %s" % row["hash_id"]})
  205. content_db.execute("UPDATE file_optional SET is_downloaded = 0, is_pinned = 0, peer = peer - 1 WHERE ?", {"site_id": site_id, "inner_path": inner_path})
  206. try:
  207. site.storage.delete(inner_path)
  208. except Exception as err:
  209. return self.response(to, {"error": "File delete error: %s" % err})
  210. site.updateWebsocket(file_delete=inner_path)
  211. if inner_path in site.content_manager.cache_is_pinned:
  212. site.content_manager.cache_is_pinned = {}
  213. self.response(to, "ok")
  214. # Limit functions
  215. def actionOptionalLimitStats(self, to):
  216. if "ADMIN" not in self.site.settings["permissions"]:
  217. return self.response(to, "Forbidden")
  218. back = {}
  219. back["limit"] = config.optional_limit
  220. back["used"] = self.site.content_manager.contents.db.getOptionalUsedBytes()
  221. back["free"] = helper.getFreeSpace()
  222. self.response(to, back)
  223. def actionOptionalLimitSet(self, to, limit):
  224. if "ADMIN" not in self.site.settings["permissions"]:
  225. return self.response(to, {"error": "Forbidden"})
  226. config.optional_limit = re.sub("\.0+$", "", limit) # Remove unnecessary digits from end
  227. config.saveValue("optional_limit", limit)
  228. self.response(to, "ok")
  229. # Distribute help functions
  230. def actionOptionalHelpList(self, to, address=None):
  231. if not address:
  232. address = self.site.address
  233. if not self.hasSitePermission(address):
  234. return self.response(to, {"error": "Forbidden"})
  235. site = self.server.sites[address]
  236. self.response(to, site.settings.get("optional_help", {}))
  237. def actionOptionalHelp(self, to, directory, title, address=None):
  238. if not address:
  239. address = self.site.address
  240. if not self.hasSitePermission(address):
  241. return self.response(to, {"error": "Forbidden"})
  242. site = self.server.sites[address]
  243. content_db = site.content_manager.contents.db
  244. site_id = content_db.site_ids[address]
  245. if "optional_help" not in site.settings:
  246. site.settings["optional_help"] = {}
  247. stats = content_db.execute(
  248. "SELECT COUNT(*) AS num, SUM(size) AS size FROM file_optional WHERE site_id = :site_id AND inner_path LIKE :inner_path",
  249. {"site_id": site_id, "inner_path": directory + "%"}
  250. ).fetchone()
  251. stats = dict(stats)
  252. if not stats["size"]:
  253. stats["size"] = 0
  254. if not stats["num"]:
  255. stats["num"] = 0
  256. self.cmd("notification", [
  257. "done",
  258. _["You started to help distribute <b>%s</b>.<br><small>Directory: %s</small>"] %
  259. (cgi.escape(title), cgi.escape(directory)),
  260. 10000
  261. ])
  262. site.settings["optional_help"][directory] = title
  263. self.response(to, dict(stats))
  264. def actionOptionalHelpRemove(self, to, directory, address=None):
  265. if not address:
  266. address = self.site.address
  267. if not self.hasSitePermission(address):
  268. return self.response(to, {"error": "Forbidden"})
  269. site = self.server.sites[address]
  270. try:
  271. del site.settings["optional_help"][directory]
  272. self.response(to, "ok")
  273. except Exception:
  274. self.response(to, {"error": "Not found"})
  275. def cbOptionalHelpAll(self, to, site, value):
  276. site.settings["autodownloadoptional"] = value
  277. self.response(to, value)
  278. def actionOptionalHelpAll(self, to, value, address=None):
  279. if not address:
  280. address = self.site.address
  281. if not self.hasSitePermission(address):
  282. return self.response(to, {"error": "Forbidden"})
  283. site = self.server.sites[address]
  284. if value:
  285. if "ADMIN" in self.site.settings["permissions"]:
  286. self.cbOptionalHelpAll(to, site, True)
  287. else:
  288. site_title = site.content_manager.contents["content.json"].get("title", address)
  289. self.cmd(
  290. "confirm",
  291. [
  292. _["Help distribute all new optional files on site <b>%s</b>"] % cgi.escape(site_title),
  293. _["Yes, I want to help!"]
  294. ],
  295. lambda (res): self.cbOptionalHelpAll(to, site, True)
  296. )
  297. else:
  298. site.settings["autodownloadoptional"] = False
  299. self.response(to, False)