| 
									
										
										
										
											2019-02-18 05:28:05 +00:00
										 |  |  | #! /usr/bin/env python3 | 
					
						
							|  |  |  | from argparse import ArgumentParser | 
					
						
							|  |  |  | from datetime import datetime | 
					
						
							| 
									
										
										
										
											2019-02-19 03:25:00 +00:00
										 |  |  | from multiprocessing import Process, Queue | 
					
						
							| 
									
										
										
										
											2019-02-19 03:33:56 +00:00
										 |  |  | from pathlib import Path | 
					
						
							| 
									
										
										
										
											2019-02-19 03:25:00 +00:00
										 |  |  | from queue import Empty | 
					
						
							| 
									
										
										
										
											2019-02-19 04:40:35 +00:00
										 |  |  | from sqlalchemy import exists, desc | 
					
						
							|  |  |  | from tempfile import TemporaryDirectory | 
					
						
							| 
									
										
										
										
											2019-02-18 05:28:05 +00:00
										 |  |  | import logging | 
					
						
							| 
									
										
										
										
											2019-02-19 03:25:00 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-19 03:33:56 +00:00
										 |  |  | from .mutation import generation, flip, seed_shell | 
					
						
							| 
									
										
										
										
											2019-02-19 04:40:35 +00:00
										 |  |  | from .orm import db_config, ScrapNode | 
					
						
							| 
									
										
										
										
											2019-02-18 23:16:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def sins(): | 
					
						
							| 
									
										
										
										
											2019-02-18 05:28:05 +00:00
										 |  |  |     now = '{0:%Y%m%dT%H%M%S}'.format(datetime.utcnow()) | 
					
						
							|  |  |  |     parser = ArgumentParser( | 
					
						
							|  |  |  |         description='position independent code (PIC) mutation experiment.') | 
					
						
							|  |  |  |     parser.add_argument('-v', '--verbose', action='count') | 
					
						
							| 
									
										
										
										
											2019-02-18 23:16:25 +00:00
										 |  |  |     parser.add_argument('-s', '--seed', help='path to PIC image.') | 
					
						
							| 
									
										
										
										
											2019-02-18 05:28:05 +00:00
										 |  |  |     parser.add_argument('-o', '--output', help='path to results directory.') | 
					
						
							| 
									
										
										
										
											2019-02-19 03:25:00 +00:00
										 |  |  |     parser.add_argument('-l', '--lineage', default=10, | 
					
						
							| 
									
										
										
										
											2019-02-19 03:33:56 +00:00
										 |  |  |                         help='max count of unsuccessful generation.') | 
					
						
							| 
									
										
										
										
											2019-02-18 05:28:05 +00:00
										 |  |  |     args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     log_level = logging.INFO | 
					
						
							|  |  |  |     log_format = logging.Formatter('%(message)s') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if args.verbose: | 
					
						
							|  |  |  |         log_level = logging.DEBUG | 
					
						
							|  |  |  |         log_format = logging.Formatter( | 
					
						
							|  |  |  |             '%(levelname)s %(filename)s:%(lineno)d\n%(message)s\n') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     logger = logging.getLogger('sins') | 
					
						
							|  |  |  |     logger.setLevel(log_level) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     stream_handler = logging.StreamHandler() | 
					
						
							|  |  |  |     stream_handler.setLevel(log_level) | 
					
						
							|  |  |  |     stream_handler.setFormatter(log_format) | 
					
						
							|  |  |  |     logger.addHandler(stream_handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if args.output: | 
					
						
							|  |  |  |         log_path = f'{args.output}/sins-{now}.log' | 
					
						
							|  |  |  |         file_handler = logging.FileHandler(log_path) | 
					
						
							|  |  |  |         file_handler.setLevel(log_level) | 
					
						
							|  |  |  |         file_handler.setFormatter(log_format) | 
					
						
							|  |  |  |         logger.addHandler(file_handler) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 23:16:25 +00:00
										 |  |  |     logger.info(now) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     seed_data = seed_shell | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if args.seed: | 
					
						
							|  |  |  |         seed = Path(args.seed) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with seed.open('rb') as seed_file: | 
					
						
							|  |  |  |             seed_data = seed_file.read() | 
					
						
							| 
									
										
										
										
											2019-02-18 05:28:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-19 04:40:35 +00:00
										 |  |  |     seed = ScrapNode(child=seed_data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     logger.info(f'seed:\n{seed}') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if args.output: | 
					
						
							|  |  |  |         db_path = Path(f'{args.output}/sins.sqlite') | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         temp_dir = TemporaryDirectory() | 
					
						
							|  |  |  |         db_path = Path(f'{temp_dir.name}/sins.sqlite') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     session = db_config(db_path) | 
					
						
							|  |  |  |     logger.info(f'db_path: {db_path}') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if args.seed: | 
					
						
							|  |  |  |         exists = session.query(ScrapNode).filter(ScrapNode.checksum == seed.checksum) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if exists: | 
					
						
							|  |  |  |             seed = exists[0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             session.add(seed) | 
					
						
							|  |  |  |             session.commit() | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         recent = session.query(ScrapNode).order_by(desc('ctime')).first() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if recent: | 
					
						
							|  |  |  |             seed = recent | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parent = seed | 
					
						
							| 
									
										
										
										
											2019-02-19 00:28:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-19 03:25:00 +00:00
										 |  |  |     queue = Queue() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while True: | 
					
						
							|  |  |  |         lineage = 0 | 
					
						
							| 
									
										
										
										
											2019-02-19 04:40:35 +00:00
										 |  |  |         scrap = flip(parent.image) | 
					
						
							| 
									
										
										
										
											2019-02-19 03:25:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while lineage < args.lineage: | 
					
						
							|  |  |  |             logger.info(f'lineage: {lineage}') | 
					
						
							|  |  |  |             result = None | 
					
						
							| 
									
										
										
										
											2019-02-19 00:28:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-19 04:40:35 +00:00
										 |  |  |             proc = Process(target=generation, args=(queue, scrap)) | 
					
						
							| 
									
										
										
										
											2019-02-19 03:25:00 +00:00
										 |  |  |             proc.start() | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 result = queue.get(timeout=1) | 
					
						
							|  |  |  |             except Empty: | 
					
						
							|  |  |  |                 lineage += 1 | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2019-02-18 05:28:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-19 03:25:00 +00:00
										 |  |  |             if not result: | 
					
						
							|  |  |  |                 lineage += 1 | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2019-02-18 23:16:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-19 04:40:35 +00:00
										 |  |  |             parent = ScrapNode(child=scrap, parent_id=parent.id) | 
					
						
							|  |  |  |             parent.length = result | 
					
						
							|  |  |  |             session.add(seed) | 
					
						
							|  |  |  |             session.commit() | 
					
						
							|  |  |  |             logger.info(f'scrap:\n{parent}') | 
					
						
							| 
									
										
										
										
											2019-02-19 03:25:00 +00:00
										 |  |  |             lineage = 0 | 
					
						
							| 
									
										
										
										
											2019-02-19 04:40:35 +00:00
										 |  |  |             scrap = flip(parent.image) |