#!/usr/bin/env python3 from PIL import Image from pprint import pprint import sys, time, os, requests, random, json, argparse parser = argparse.ArgumentParser(description='Unknown program that breaks in strange ways.') parser.add_argument("--download", help="Download Images", action="store_true") parser.add_argument("JSON", type=str, nargs="?", help="Filename to save results", default="test.js") args = parser.parse_args() pprint(args) # Should we spend the time to download image, and process it? (True = Yes, False = No) # DOWNLOAD = False DOWNLOAD = args.download # Should we add the JSON in a file? (True is filename, False = do not do) # JSONME = 'test.js' JSONME = args.JSON # NOTE: To begin the insert of the JSONIFIED image and word its # // T # A JS comment with a uppercase T # To stop its # // t # A JS comment with a lowercase t # httpbin.org/headers sess = requests.Session() head = {'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'} sess.headers.update(head) ON = 'X' # Dark pixel in an image OFF = '.' # Light pixel in an image DIR = 'data' # Data directory name, do we really need this? Is it really going to change? INTENSITY = 75 # How bright does something have to be to trigger it being a dark or light pixel? # Looks like around 75 removes the extra stuff that s0urce.io does to prevent it from being just matching images. # Globals, Yuck! im = '' pix = [0, 0] size = [0, 0] pal = '' # Check the environment, do we have all that we need? if not os.path.exists('in'): os.mkdir('in') if not os.path.exists('data'): os.mkdir('data') def download(howhard, index): global sess """ Download an image based upon how hard it is. You print out the return value. Are you returning anything useful?? (Ah, yeah I could just have it print out the status code when there is a error, Which would get rid of the need for it to return anything) """ r = sess.get(f'http://s0urce.io/client/img/word/{howhard}/{index}') if(r.status_code == 200): with open(f'in/{howhard}_{index}.png', 'wb') as f: f.write(r.content) else: # We did not get a 200 Okay, log this... Hmm maybe we need to make a log file? #print( f'{howhard}_{index}.png ' + str(r.status_code) ) raise ConnectionError('http://s0urce.io/client/img/word/{0}/{1} returned status_code {2}'.format(howhard, index, r.status_code)) def img_point(pix, x, y): """ img_point, returns a pixel of an image, given the x and y on the image. """ return pix[x, y] def img_avg(im, pal, pix, x, y): """ img_avg, returns the average brightness 0-255, given the x and y on the image calls img_point, to get the individual rgb values to calculate, brightness. """ rgb = img_point(pix, x,y) if(im.mode == 'P'): rgb = pal[rgb*3:(rgb+1)*3] if(im.mode == 'I'): return rgb >> 8 return int( ( rgb[0] + rgb[1] + rgb[2] ) / 3 ) def is_set(im, pal, pix, x, y): global INTENSITY """ is_set, returns True or False of calculating, the brightness of the given point on a image, compared to given intensity. True means the brightness at the given x and y, is Less Than which means its dark. False means the brightness at the given x and y, is Greater Than which means its bright. """ avg = img_avg(im, pal, pix, x,y) return (avg < INTENSITY) def scan_img(im, pal, pix, size): """ scan_img, looks at a image and looks for dark pixels, if it is a dark pixel record the number and resize the, returned values to show where the most dark pixels on the, image are located. given image size. returns start x, y and end x, y and total number of dark pixels. """ total = 0 sx = size[0]; ex = 0 sy = size[1]; ey = 0 for y in range( 0,size[1] ): for x in range( 0,size[0] ): pnt_is = is_set(im, pal, pix, x,y) if (pnt_is): total += 1 if x < sx: sx = x if x > ex: ex = x if y < sy: sy = y if y > ey: ey = y #print (sx,ex,sy,ey) # give us a little border to work with if sx > 0: sx -= 1 if ex < size[0]: ex += 1 if sy > 0: sy -= 1 if ey < size[1]: ey += 1 #print (sx,ex,sy,ey) return(sx,sy,ex,ey,total) def output_image(im, pal, pix, size): """ For the size of the area we have reduced down to where the majority of dark pixels, are located, store all that into a list and return the list. given image, palette, pixel for function passing. returns multiple strings in a list that are edited to use characters to represent, the dark and light pixels of the image. """ result = [] ex = size[0]; sx = 0 ey = size[1]; sy = 0 for y in range(sy,ey): s = '' for x in range(sx,ex): if is_set(im, pal, pix, x,y): s += ON else: s += OFF result.append(s) return result def run(difficult): """ 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 ) those components do the following... (Each category has around 70 items so we standardize on 70, but ) (not all of the categories have 70 and thus we print a File does not exist) We open and load the image, and get it's size, then we scan_img for dark and light pixels, <-- This narrows the image down to just the majority of dark pixels then from that we output the image line by line onto the screen after it has been output_image d into list form, Where we ask the user what the word is, and after that we save all that to a file in the data directory. """ for x in range(0, 70): fname = f'in/{difficult}_{x}.png' if not os.path.exists(fname): print("Could not find '{0}'".format(fname)) continue print(f"Loading: {fname}") im = Image.open(fname) pix = im.load() size = im.size print(f"Size: {size[0]} x {size[1]}") pal = im.getpalette() sx = 0 ex = size[0] sy = 0 ey = size[1] total = 0 sx, sy, ex, ey, total = scan_img(im, pal, pix, size) print(f"Chars within ({sx}, {sy}) - ({ex}, {ey}) total {total} pixels") img_s = output_image(im, pal, pix, size) for l in img_s: print(l) word = input('Word: ') with open(f'{DIR}/{difficult}_{x}.txt', 'w') as f: f.write('{0}\n'.format(word)) print(f"Image saved to '{DIR}/{difficult}_{x}.txt' in byte string") # os.remove(f'{fname}') # Grr No bad bean, keep file for error checking # print(f"File '{fname}' automatically removed") # Now to call all the previous functions if (DOWNLOAD == True): print("Downloading s0urce.io Words") print("EASY") # time.sleep(5) for e in range(0, 62): download('e', e) # time.sleep(random.randint(10, 15)) print("MEDIUM") # time.sleep(5) for m in range(0, 66): download('m', m) # time.sleep(random.randint(10, 15)) print("HARD") # time.sleep(5) for h in range(0, 55): download('h', h) # time.sleep(random.randint(10, 15)) # Img Processing run('e') # Answer the questions run('m') run('h') # ---------------------------------------------------------------------------------------- # All below was in a seperate dataJS.py file... but now I have fixed it so it's 1 script! # Do we really need to worry about all this right now? (I think we have enough bugs to begin with.) JSONME = 'false' # Do not execute if (JSONME.lower() != 'false'): print("Now exporting to JSON") print(f"Targeting file: '{JSONME}'") time.sleep(5) def test(t): global DIR """ given the filename, we read it and add it to a list and return the list. """ fname = f'{DIR}/{t}.txt' r = [] try: with open(fname, 'r') as f: for l in f: r.append(l.strip()) return r except FileNotFoundError: return None def insertJS(item): global JSON """ Edits the file given and adds the JSONIFIED item to the file between 2 indicators, // T and // t In between the T and t will be replaced with the item. """ item = json.dumps(item) item = f'{item},' r = [] try: with open(f'{JSONME}', 'r') as f: for l in f: if l != '': r.append(l.strip('\n')) else: r.append('') except FileNotFoundError: print(f"File {JSONME} Not Found!") sys.exit() c = 0 for e in r: if('// T' == e): temp = r[c+1] del r[c+1] r.insert(c+1, item) r.insert(c+2, temp) elif('// t' == e): break c += 1 with open(f'{JSONME}', 'w') as f: for e in r: f.write(f'{e}\n') for x in range(0, 183): te = test(x) if(te != None): word = te insertJS(word) # Regardless what we did let the user know we at least ran and we are now done print("Complete")