#!/usr/bin/env python
from argparse import ArgumentParser

from configparser import ConfigParser
import os, imaplib, email, toml, re
from glob import glob
from datetime import datetime, timedelta

from pypromice.tx import getMail, L0tx, sortLines


def parse_arguments():
    parser = ArgumentParser(description="AWS L0 transmission fetcher")       
    parser.add_argument('-a', '--account', default=None, type=str, required=True, help='Email account .ini file')
    parser.add_argument('-p', '--password', default=None, type=str, required=True, help='Email credentials .ini file')                      
    parser.add_argument('-o', '--outpath', default=None, type=str, required=False, help='Path where to write output (if given)')   
    parser.add_argument('-c', '--config', default=None, type=str, required=True, help='Directory to config .toml files')         
    parser.add_argument('-f', '--formats', default=None, type=str, required=False, help='Path to Payload format .csv file')
    parser.add_argument('-t', '--types', default=None, type=str, required=False, help='Path to Payload type .csv file')  	
    parser.add_argument('-u', '--uid', default=None, type=str, required=True, help='Last AWS uid .ini file')	        
    args = parser.parse_args()
    return args


if __name__ == '__main__':
    """Executed from the command line"""
    args = parse_arguments()
    toml_path = os.path.join(args.config, '*.toml')
    toml_list = glob(toml_path)
    
    aws={}
    for t in toml_list:
        conf = toml.load(t)
        count=1
        for m in conf['modem']:
            name = str(conf['station_id']) + '_' + m[0] + '_' + str(count)
   	        
            if len(m[1:])==1:
                   aws[name] = [datetime.strptime(m[1],'%Y-%m-%d %H:%M:%S'), datetime.now() + timedelta(hours=3)]
            elif len(m[1:])==2:
                   aws[name] = [datetime.strptime(m[1],'%Y-%m-%d %H:%M:%S'), datetime.strptime(m[2], '%Y-%m-%d %H:%M:%S')]
       	    count+=1

 	#----------------------------------

 	# Set payload formatter paths
    formatter_file = args.formats
    type_file = args.types

 	# Set credential paths
    accounts_file = args.account
    credentials_file = args.password 

 	# Set last aws uid path
 	# last_uid = 1000000
    uid_file = args.uid
 
 	# Set output file directory
    out_dir = args.outpath
    if out_dir is not None:
        if not os.path.exists(out_dir):
            os.mkdir(out_dir)

 	#----------------------------------

 	# Define accounts and credentials ini file paths
    accounts_ini = ConfigParser()
    accounts_ini.read_file(open(accounts_file))
    accounts_ini.read(credentials_file)

 	# Get last email uid
    with open(uid_file, 'r') as last_uid_f:
        last_uid = int(last_uid_f.readline())
  
 	# Get credentials
    account = accounts_ini.get('aws', 'account')
    server = accounts_ini.get('aws', 'server')
    port = accounts_ini.getint('aws', 'port')    
    password = accounts_ini.get('aws', 'password')
    if not password:
        password = input('password for AWS email account: ')
    print('AWS data from server %s, account %s' %(server, account))

 	#----------------------------------

 	# Log in to email server
    mail_server = imaplib.IMAP4_SSL(server, port)
    typ, accountDetails = mail_server.login(account, password) 	
    if typ != 'OK':
        print('Not able to sign in!')
        raise

 	# Grab new emails
    result, data = mail_server.select(mailbox='"[Gmail]/All Mail"', readonly=True)
    print('mailbox contains %s messages' %data[0])

 	#----------------------------------

 	# Get L0tx datalines from email transmissions
    uid=last_uid # initialize using last_uid. If no messages, this will be written back to ini file.
    for uid, mail in getMail(mail_server, last_uid=last_uid):
        message = email.message_from_string(mail)
        
        try:
            imei, = re.findall(r'[0-9]+', message.get_all('subject')[0])
            d = datetime.strptime(message.get_all('date')[0], '%a, %d %b %Y %H:%M:%S %Z')
        except:
            imei = None
            d = None
        for k,v in aws.items():
            if str(imei) in k:
                if v[0] < d < v[1]:
                    print(f'AWS message for {k}.txt, {d.strftime("%Y-%m-%d %H:%M:%S")}')
                    l0 = L0tx(message, formatter_file, type_file)

                    if l0.msg:
                        body = l0.getEmailBody()
                        if 'Lat' in body[0] and 'Long' in body[0]:
                            lat_idx = body[0].split().index('Lat') + 2
                            lon_idx = lat_pos = body[0].split().index('Long') + 2
                            lat = body[0].split()[lat_idx]
                            lon = body[0].split()[lon_idx]
                            # append lat and lon to end of message
                            l0.msg = l0.msg + ',{},{}'.format(lat,lon)
                        print(l0.msg)

                        if out_dir is not None:
                            out_fn = str(k) + str(l0.flag) + '.txt'
                            out_path = os.sep.join((out_dir, out_fn))
                            print(f'Writing to {out_fn}')
                        		
                            with open(out_path, mode='a') as out_f:
                                out_f.write(l0.msg + '\n')    

 	#----------------------------------
 	
    if out_dir is not None:
        # Sort L0tx files and add tails    
        for f in glob(out_dir+'/*.txt'):
        
        	# Sort lines in L0tx file and remove duplicates
            in_dirn, in_fn = os.path.split(f)    
            out_fn = 'sorted_' + in_fn
            out_pn = os.sep.join((in_dirn, out_fn))
            sortLines(f, out_pn)
        
        	## Generate tail files
        	# out_dir = os.sep.join((in_dirn, 'tails')) 
        	# if not os.path.exists(out_dir):
        	#     os.mkdir(out_dir)
        	# imei = in_fn.split('.txt')[0].split('_')[1].split('-')[0]        
        	# name = imei_names.get(imei, 'UNKNOWN')
        	# addTail(f, out_dir, name)

    #---------------------------------

 	# Close mail server if open
    if 'mail_server' in locals():
        print(f'\nClosing {account}')
        mail_server.close()
        resp = mail_server.logout()
        assert resp[0].upper() == 'BYE'

    #---------------------------------

 	# Write last aws uid to ini file
    try:
        with open(uid_file, 'w') as last_uid_f:
            last_uid_f.write(str(uid))
    except:
        print(f'Could not write last uid {uid} to {uid_file}')
    
    print('Finished')
    
else:
    """Executed on import"""
    pass
