dataLoad.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428
  1. #!/usr/bin/env python3
  2. from PIL import Image
  3. from pprint import pprint
  4. import sys
  5. import time
  6. import os
  7. import requests
  8. import random
  9. import json
  10. import argparse
  11. import subprocess
  12. import imager
  13. parser = argparse.ArgumentParser(description="S0urce.io utility program.")
  14. parser.add_argument("--download", help="Download Images", action="store_true")
  15. parser.add_argument("--train", help="Convert Images to Text", action="store_true")
  16. parser.add_argument("--update", help="Update s0urce.js script", action="store_true")
  17. parser.add_argument(
  18. "JSON", type=str, nargs="?", help="Filename to save results", default="test.js"
  19. )
  20. args = parser.parse_args()
  21. # pprint(args)
  22. # Should we add the JSON in a file? (True is filename, False = do not do)
  23. # JSONME = 'test.js'
  24. JSONME = args.JSON
  25. # NOTE: To begin the insert of the JSONIFIED image and word its
  26. # // T
  27. # A JS comment with a uppercase T
  28. # To stop its
  29. # // t
  30. # A JS comment with a lowercase t
  31. # httpbin.org/headers
  32. sess = requests.Session()
  33. head = {
  34. "User-Agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/39.0.2171.95 Safari/537.36"
  35. }
  36. sess.headers.update(head)
  37. ON = "X" # Dark pixel in an image
  38. OFF = "." # Light pixel in an image
  39. DIR = (
  40. "data"
  41. ) # Data directory name, do we really need this? Is it really going to change?
  42. INTENSITY = (
  43. 75
  44. ) # How bright does something have to be to trigger it being a dark or light pixel?
  45. # Looks like around 75 removes the extra stuff that s0urce.io does to prevent it from being just matching images.
  46. GREEN_DIFF = 10
  47. # How much brighter the green channel must be (compared to the others),
  48. # to be called green.
  49. VALID_WORDS = {
  50. "constructor", "info", "anon", "send", "com", "root", "port", "val",
  51. "add", "ghost", "net", "http", "status", "syscall", "part", "delete",
  52. "datatype", "loadbytes", "setping", "size", "system", "setstats",
  53. "join", "socket", "signal", "dir", "accountname", "decryptfile",
  54. "intel", "xml", "connect", "sizeof", "writefile", "call", "reset",
  55. "global", "user", "add", "remove", "count", "set", "loop", "num",
  56. "client", "file", "channel", "right", "stat", "emit", "handle",
  57. "buffer", "mysql", "write", "type", "list", "temp", "getfile",
  58. "thread", "decrypt", "poly", "setcookie", "domain", "length",
  59. "gridwidth", "upload", "get", "generatecodepack", "data",
  60. "process", "download", "proxy", "fillgrid", "bit", "encryptfile",
  61. "host", "ping", "event", "url", "load", "key", "changepassword",
  62. "bufferpingset", "getfirewallchannel", "getinfo", "getping", "pass",
  63. "newserver", "username", "generate", "userport", "init", "net",
  64. "left", "point", "cookies", "protocol", "responder", "getkey",
  65. "hostserver", "eventtype", "gridheight", "server", "setport",
  66. "getpass", "loadloggedpassword", "destroybatch", "getxmlprotocol",
  67. "channelsetpackage", "batchallfiles", "module", "response",
  68. "serverproxy", "filetype", "urlcheck", "config", "number",
  69. "ghostfilesystem", "disconnectserver", "emitconfiglist",
  70. "dodecahedron", "eventlistdir", "systemportkey", "setnewproxy",
  71. "createnewsocket", "changeusername", "tempdatapass", "blockthreat",
  72. "statusofprocess", "patcheventlog", "newline", "dir", "bytes",
  73. "findpackage", "package", "encode", "joinnetworkclient",
  74. "rootcookieset", "callmodule", "sizeofhexagon", "createfilethread",
  75. "includedirectory", "loadregisterlist", "encryptunpackedbatch",
  76. "getpartoffile", "getdatapassword", "create2axisvector",
  77. "create3axisvector", "disconnectchannel", "setnewid", "hexagon",
  78. "account", "removenewcookie", "getid", "encodenewfolder",
  79. "sendintelpass", "getlog", "command", "threat", "userid",
  80. "wordcounter", "removeoldcookie", "hostnewserver", "disconnect",
  81. "listconfig", "newhost", "createnewpackage", "loadaltevent", "log",
  82. "filedir", "fileexpresslog", "decryptdatabatch", "mergesocket",
  83. "unpacktmpfile", "uploaduserstats", "getmysqldomain",
  84. "checkhttptype", "encrypt", "vector", "httpbuffersize",
  85. "systemgridtype", "password", "respondertimeout", "deleteallids",
  86. "exportconfigpackage", "export"
  87. }
  88. # Check the environment, do we have all that we need?
  89. if not os.path.exists("images"):
  90. os.mkdir("images")
  91. if not os.path.exists("data"):
  92. os.mkdir("data")
  93. if not os.path.exists("words.yml"):
  94. with open('words.yml', 'w') as f: # Create a empty yaml file
  95. f.write('')
  96. def image_filename(difficulty, index):
  97. return f"images/{difficulty}_{index}.png"
  98. def cleaned_filename(difficulty, index):
  99. return f"images/{difficulty}_{index}_clean.png"
  100. def cleaner_filename(difficulty, index):
  101. return f"images/{difficulty}_{index}_cleaner.png"
  102. def download(howhard, index):
  103. global sess
  104. """
  105. Download an image based upon how hard it is.
  106. On success, it saves the image file.
  107. Failure raises ConnectionError.
  108. Don't leave stale cleaned images around.
  109. """
  110. r = sess.get(f"http://s0urce.io/client/img/word/{howhard}/{index}")
  111. if r.status_code == 200:
  112. # DRY
  113. with open( image_filename(howhard, index), "wb") as f:
  114. f.write(r.content)
  115. # cleaned images? we need to delete & regenerate those.
  116. cleaned = cleaned_filename(howhard, index)
  117. if os.path.exists(cleaned):
  118. os.remove(cleaned)
  119. else:
  120. # We did not get a 200 Okay, log this... Hmm maybe we need to make a log file?
  121. # print( f'{howhard}_{index}.png ' + str(r.status_code) )
  122. raise ConnectionError(
  123. "http://s0urce.io/client/img/word/{0}/{1} returned status_code {2}".format(
  124. howhard, index, r.status_code
  125. )
  126. )
  127. def img_point(pix, x, y):
  128. """
  129. img_point, returns a pixel of an image,
  130. given the x and y on the image.
  131. """
  132. return pix[x, y]
  133. def img_avg(pix, x, y):
  134. """
  135. img_avg, returns the average brightness 0-255,
  136. given pixel, and the x and y on the image calls img_point,
  137. to get the individual rgb values to calculate,
  138. brightness. (Grey scale)
  139. """
  140. rgb = img_point(pix, x, y)
  141. # if(im.mode == 'P'):
  142. # rgb = pal[rgb*3:(rgb+1)*3]
  143. # if(im.mode == 'I'):
  144. # return rgb >> 8
  145. return int((rgb[0] + rgb[1] + rgb[2]) / 3)
  146. def is_set(pix, x, y):
  147. global INTENSITY
  148. """
  149. is_set, returns True or False of calculating,
  150. the brightness of the given point on a image,
  151. compared to given intensity.
  152. True means the brightness at the given x and y,
  153. is Less Than which means its dark.
  154. False means the brightness at the given x and y,
  155. is Greater Than which means its bright. (Grey Scale)
  156. """
  157. avg = img_avg(pix, x, y)
  158. return avg < INTENSITY
  159. def is_green(pix, x, y):
  160. """
  161. Is this pixel Green?
  162. """
  163. (red, green, blue, _) = img_point(pix, x, y)
  164. # Find the difference between green and the other values.
  165. other = red
  166. if blue > other:
  167. other = blue
  168. diff = green - other
  169. return diff > GREEN_DIFF
  170. def scan_img(pix, size):
  171. """
  172. scan_img, looks at a image and looks for dark pixels,
  173. if it is a dark pixel record the number and resize the,
  174. returned values to show where the most dark pixels on the,
  175. image are located. (Grey Scale)
  176. given pixel, and image size.
  177. returns start x, y and end x, y and total number of dark pixels.
  178. """
  179. total = 0
  180. sx = size[0]
  181. ex = 0
  182. sy = size[1]
  183. ey = 0
  184. for y in range(0, size[1]):
  185. for x in range(0, size[0]):
  186. pnt_is = is_set(pix, x, y)
  187. if pnt_is:
  188. total += 1
  189. if x < sx:
  190. sx = x
  191. if x > ex:
  192. ex = x
  193. if y < sy:
  194. sy = y
  195. if y > ey:
  196. ey = y
  197. # print (sx,ex,sy,ey)
  198. # give us a little border to work with
  199. if sx > 0:
  200. sx -= 1
  201. if ex < size[0]:
  202. ex += 1
  203. if sy > 0:
  204. sy -= 1
  205. if ey < size[1]:
  206. ey += 1
  207. # print (sx,ex,sy,ey)
  208. return (sx, sy, ex, ey, total)
  209. def output_image(pix, size):
  210. """
  211. For the size of the area we have reduced down to where the majority of dark pixels,
  212. are located, store all that into a list and return the list.
  213. given pixel for function passing.
  214. returns multiple strings in a list that are edited to use characters to represent,
  215. the dark and light pixels of the image. (Grey Scale)
  216. """
  217. result = []
  218. ex = size[0]
  219. sx = 0
  220. ey = size[1]
  221. sy = 0
  222. for y in range(sy, ey):
  223. s = ""
  224. for x in range(sx, ex):
  225. # if is_set(pix, x, y):
  226. if not is_green(pix, x, y):
  227. s += ON
  228. else:
  229. s += OFF
  230. result.append(s)
  231. return result
  232. def run(difficult, index):
  233. """
  234. run, represents a single execution of components to the image, (Actuall we do it 1 category at a time instead of just 1 single execution )
  235. those components do the following... (Each category has around 70 items so we standardize on 70, but )
  236. (not all of the categories have 70 and thus we print a File does not exist)
  237. We open and load the image, and get it's size,
  238. then we scan_img for dark and light pixels, <-- This narrows the image down to just the majority of dark pixels
  239. then from that we output the image line by line onto the screen after it has been output_image d into list form,
  240. Where we ask the user what the word is, and after that we save all that to a file in the data directory.
  241. """
  242. for x in range(0, 70):
  243. fname = image_filename(difficult, x)
  244. if not os.path.exists(fname):
  245. # print("Could not find '{0}'".format(fname))
  246. # continue
  247. # We've reached the end, so stop looking. :P
  248. break
  249. print(f"Loading: {fname}")
  250. im = Image.open(fname)
  251. pix = im.load()
  252. size = im.size
  253. print(f"Size: {size[0]} x {size[1]}")
  254. pal = im.getpalette()
  255. sx = 0
  256. ex = size[0]
  257. sy = 0
  258. ey = size[1]
  259. total = 0
  260. sx, sy, ex, ey, total = scan_img(pix, size)
  261. print(f"Chars within ({sx}, {sy}) - ({ex}, {ey}) total {total} pixels")
  262. img_s = output_image(pix, size)
  263. for l in img_s:
  264. print(l)
  265. word = input("Word: ")
  266. # Returns word so it can be stored in dictonary
  267. return word
  268. #print(f"Image saved to '{DIR}/{difficult}_{x}.txt' in byte string")
  269. # os.remove(f'{fname}') # Grr No bad bean, keep file for error checking
  270. # print(f"File '{fname}' automatically removed")
  271. key_word = {}
  272. def autotrain(difficult):
  273. """
  274. run, represents a single execution of components to the image, (Actuall we do it 1 category at a time instead of just 1 single execution )
  275. those components do the following... (Each category has around 70 items so we standardize on 70, but )
  276. (not all of the categories have 70 and thus we print a File does not exist)
  277. We open and load the image, and get it's size,
  278. then we scan_img for dark and light pixels, <-- This narrows the image down to just the majority of dark pixels
  279. then from that we output the image line by line onto the screen after it has been output_image d into list form,
  280. Where we ask the user what the word is, and after that we save all that to a file in the data directory.
  281. """
  282. for x in range(0, 70):
  283. fname = image_filename(difficult, x)
  284. if not os.path.exists(fname):
  285. break
  286. # print("Could not find '{0}'".format(fname))
  287. # continue
  288. cleaned = cleaned_filename(difficult, x)
  289. if not os.path.exists(cleaned):
  290. imager.image_cleaner(fname, cleaned)
  291. print(f"Loading: {cleaned} ", end='')
  292. fileout = "data/{0}_{1}".format(difficult, x)
  293. output = subprocess.run(
  294. ["tesseract", cleaned, fileout],
  295. stderr=subprocess.DEVNULL,
  296. # capture_output=False,
  297. shell=False,
  298. )
  299. with open(fileout + ".txt", "r") as fp:
  300. word = fp.read().strip().lower()
  301. if (word != '') and (word in VALID_WORDS):
  302. key_word[f'{difficult}_{x}'] = word
  303. print(word)
  304. else:
  305. print("UNKNOWN", word)
  306. # Output the image
  307. im = Image.open(fname)
  308. imager.output_image(im)
  309. # pix = im.load()
  310. # size = im.size
  311. # img_s = output_image(pix, size)
  312. # for l in img_s:
  313. # print(l)
  314. key_word[f'{difficult}_{x}'] = input("Word: ")
  315. # Now to call all the previous functions
  316. if args.download:
  317. print("Downloading s0urce.io Words")
  318. # smaller is better, and cleaner.
  319. tofetch = { 'e': 62, 'm': 66, 'h': 55 }
  320. for d, max in tofetch.items():
  321. print(d.upper())
  322. for i in range(0, max):
  323. download(d, i)
  324. # time.sleep(random.randint(10, 15))
  325. if args.train:
  326. # Img Processing: Run thru every single category and every single word
  327. for level in ["e", "m", "h"]:
  328. autotrain(level)
  329. with open(args.JSON, 'w') as fp:
  330. json.dump(key_word, fp, sort_keys=True, indent=2)
  331. if args.update:
  332. with open(args.JSON, 'r') as fp:
  333. key_word = json.load(fp)
  334. # update the s0urce.js script
  335. filename = 's0urce.js'
  336. with open(filename, 'r') as fp:
  337. lines = fp.readlines()
  338. # Lines are now in memory. Time to update!
  339. for i in range(0, len(lines)):
  340. if 'http://s0urce.io/client/img/word/' in lines[i]:
  341. # This is a target line, so:
  342. l = lines[i].strip().strip(':').strip('"')
  343. # gets parts of the path
  344. parts = l.split('/')
  345. # get difficulty and index
  346. dif = parts[-2]
  347. index = parts[-1]
  348. # build the key -- get the word
  349. key = f'{dif}_{index}'
  350. pprint(parts)
  351. pprint(key)
  352. word = key_word[key]
  353. lines[i+1] = f' form.value = "{word}";' + "\n" # break;\n" # You may need it... or may not.
  354. with open(filename, 'w') as fp:
  355. for line in lines:
  356. fp.write(line)
  357. # Regardless what we did let the user know we at least ran and we are now done
  358. print("Complete")