#functions for using lucent to get visualizations
from lucent_circuit.modelzoo.util import get_model_layers
from lucent_circuit.optvis import render, param, transform, objectives

import inspect
import time
import pandas as pd

from circuit_explorer.utils import *
import sys
#sys.path.insert(0, os.path.abspath('../prep_model_scripts/'))
from circuit_explorer.dissected_Conv2d import *
from circuit_explorer import root_path
from circuit_explorer.utils import convert_relu_layers
import random

from subprocess import call

from circuit_explorer.receptive_fields import receptive_field, receptive_field_for_unit
from math import ceil, floor
from copy import deepcopy
from circuit_explorer.dissected_Conv2d import dissect_model
from circuit_explorer.utils import convert_relu_layers
from math import floor
import numpy as np


# def gen_visualization(model,image_name,objective,parametrizer,optimizer,transforms,image_size,neuron,params):
# 	if neuron:
# 		full_image_path = params['prepped_model_path']+'/visualizations/images/neuron/'+image_name
# 	else:
# 		full_image_path = params['prepped_model_path']+'/visualizations/images/channel/'+image_name
# 	if parametrizer is None:
# 		parametrizer = lambda: param.image(image_size)
# 	print('generating featviz with objective: %s'%str(objective))
# 	_ = render.render_vis(model, objective, parametrizer, optimizer, transforms=transforms,save_image=True,image_name=full_image_path, show_inline=True)


def gen_objective_str(targetid,model,params, neuron=False):
	print('generating feature_viz objective string for %s'%targetid)
	if '-' in str(targetid):
		within_id = edgename_2_singlenum(model,targetid,params)
		layer_name = nodeid_2_perlayerid(targetid.split('-')[1],params)[2]
		if not neuron:
			return layer_name+'_preadd_conv:'+str(within_id)
		else:
			return objectives.neuron(layer_name+'_preadd_conv', int(within_id))
	else:
		layer, within_id, layer_name = nodeid_2_perlayerid(targetid,params)
		if not neuron:
			return layer_name+':'+str(within_id)
		else:
			return objectives.neuron(layer_name,int(within_id))


#possibility of neuron
def gen_objective(targetid,model,params,neuron=False):
	print('generating feature_viz objective string for %s'%targetid)
	if '-' in targetid:
		within_id = edgename_2_singlenum(model,targetid,params)
		layer_name = nodeid_2_perlayerid(targetid.split('-')[1],params)[2]
		if not neuron:
			#return layer_name+'_preadd_conv:'+str(within_id)
			return objectives.channel(layer_name+'_preadd_conv', int(within_id))
		else:
			return objectives.neuron(layer_name+'_preadd_conv', int(within_id))

	else:
		layer, within_id, layer_name = nodeid_2_perlayerid(targetid,params)
		if not neuron:
			#return layer_name+':'+str(within_id)
			return objectives.channel(layer_name,int(within_id))
		else:
			return objectives.neuron(layer_name,int(within_id))



def object_2_str(obj,prefix):
	if isinstance(obj,str):
		return obj
	if isinstance(obj,list):
		return obj
	if obj is None:
		return('none')
	else:
		s= inspect.getsource(obj).replace(' ','').replace(prefix,'').strip()
		return inspect.getsource(obj)
	

# def fetch_deepviz_img(model,targetid,params):
# 	model = set_across_model(model,'target_node',None)
# 	objective_str = gen_objective_str(targetid,model,params)
# 	neuron = params['deepviz_neuron']
# 	objective = gen_objective(targetid,model,params,neuron=neuron)
# 	file_path = params['prepped_model_path']+'/visualizations/images.csv'
# 	parametrizer = params['deepviz_param']
# 	optimizer = params['deepviz_optim']
# 	transforms = params['deepviz_transforms']
# 	image_size = params['deepviz_image_size']

# 	param_str = object_2_str(parametrizer,"params['deepviz_param']=")
# 	optimizer_str = object_2_str(optimizer,"params['deepviz_optim']=")
# 	transforms_str = object_2_str(transforms,"params['deepviz_transforms']=")
# 	df = pd.read_csv(file_path,dtype=str)
# 	df_sel = df.loc[(df['targetid'] == str(targetid)) & (df['objective'] == objective_str) & (df['parametrizer'] == param_str) & (df['optimizer'] == optimizer_str) & (df['transforms'] == transforms_str) & (df['neuron'] == str(neuron))]
# 	if len(df_sel) == 0:
# 		print('deepviz image not found for %s, generating . . .'%targetid)
# 		#image_name = 'deepviz_'+str(targetid)+'_'+objective+'_'+str(time.time())+'.jpg'
# 		image_name = str(targetid)+'_'+objective_str+'_'+str(time.time())+'.jpg'
# 		gen_visualization(model,image_name,objective,parametrizer,optimizer,transforms,image_size,neuron,params) 
# 		with open(file_path, 'a') as csv:
# 			csv.write(','.join([image_name,str(targetid),objective_str,param_str,optimizer_str,transforms_str,str(neuron)])+'\n')
# 	else:
# 		print('found pre-generated image')
# 		image_name = df_sel.iloc[0]['image_name']  

# 	if neuron:  
# 		return 'neuron/'+image_name
# 	else:
# 		return 'channel/'+image_name

# def fetch_deepviz_img_for_node_inputs(model,edgeid,params):
# 	model = set_across_model(model,'target_node',None)
# 	objective_str = gen_objective_str(edgeid,model,params)
# 	file_path = params['prepped_model_path']+'/visualizations/images.csv'
# 	parametrizer = params['deepviz_param']
# 	optimizer = params['deepviz_optim']
# 	transforms = params['deepviz_transforms']
# 	image_size = params['deepviz_image_size']
# 	neuron = params['deepviz_neuron']

# 	objective = gen_objective(edgeid,model,params,neuron)
# 	param_str = object_2_str(parametrizer,"params['deepviz_param']=")
# 	optimizer_str = object_2_str(optimizer,"params['deepviz_optim']=")
# 	transforms_str = object_2_str(transforms,"params['deepviz_transforms']=")
# 	df = pd.read_csv(file_path,dtype=str)

# 	nodeid = edgeid.split('-')[0]
# 	if 'r' in edgeid or 'g' in edgeid or 'b' in edgeid or 'gs' in edgeid:
# 		df_sel = df.loc[(df['targetid'] == str(edgeid)) & (df['objective'] == objective_str) & (df['parametrizer'] == param_str) & (df['optimizer'] == optimizer_str) & (df['transforms'] == transforms_str) & (df['neuron'] == str(neuron))]
# 		if len(df_sel) == 0:
# 			print('deepviz image not found for node input %s, generating . . .'%edgeid)
# 			#image_name = 'deepviz_'+str(targetid)+'_'+objective+'_'+str(time.time())+'.jpg'
# 			image_name = str(edgeid)+'_'+objective_str+'_'+str(time.time())+'.jpg'
# 			gen_visualization(model,image_name,objective,parametrizer,optimizer,transforms,image_size,neuron,params) 
# 			with open(file_path, 'a') as csv:
# 				csv.write(','.join([image_name,str(edgeid),objective_str,param_str,optimizer_str,transforms_str,str(neuron)])+'\n') 
# 		else:       
# 			print('found pre-generated edge image')
# 			image_name = df_sel.iloc[0]['image_name']
# 	else:
# 		objective = gen_objective_str(nodeid,model,params)
# 		df_sel = df.loc[(df['targetid'] == str(nodeid)) & (df['objective'] == objective) & (df['parametrizer'] == param_str) & (df['optimizer'] == optimizer_str) & (df['transforms'] == transforms_str) & (df['transforms'] == transforms_str) & (df['neuron'] == str(neuron))]
# 		if len(df_sel) == 0:
# 			print('no deepviz image found for %s! generating now, but you should pregenerate these!'%str(nodeid))
# 			image_name = str(nodeid)+'_'+objective_str+'_'+str(time.time())+'.jpg'
# 			gen_visualization(model,image_name,objective,parametrizer,optimizer,transforms,image_size,neuron,params) 
# 		else:
# 			print('found pregenerated node image')
# 			image_name = df_sel.iloc[0]['image_name'] 

# 	if neuron:  
# 		return 'neuron/'+image_name
# 	else:
# 		return 'channel/'+image_name



def fetch_deepviz_img_for_circuit(circuit,layer_name,within_id,targetid,device='cuda',viz_folder=None,neuron_viz=True,recep_field=True):

	objective_str = layer_name+':'+str(targetid)

	#generate objective
	#objective = gen_objective(targetid,model,params,neuron=neuron)
	print('generating feature_viz objective string for %s'%targetid)

	if not neuron_viz:
			#return layer_name+':'+str(within_id)
		objective = objectives.channel(layer_name,int(within_id))
	else:
		objective = objectives.neuron(layer_name,int(within_id))


	if viz_folder is None:
		viz_folder = os.path.join(root_path,'visualization_cache','temp')
	file_path = viz_folder+'/images.csv'
	if not os.path.exists(file_path):
		call('touch %s'%file_path,shell=True)
		f = open(file_path, "a")
		f.write('image_name,targetid,objective,parametrizer,optimizer,transforms,neuron,recep_field')
		f.close()

	parametrizer = None
	optimizer = None
	transforms = None
	image_size = 224

	param_str = object_2_str(parametrizer,"parametrizer =")
	optimizer_str = object_2_str(optimizer,"optimizer =")
	transforms_str = object_2_str(transforms,"transforms =")
	df = pd.read_csv(file_path,dtype=str)
	df_sel = df.loc[(df['targetid'] == str(targetid)) & (df['objective'] == objective_str) & (df['parametrizer'] == param_str) & (df['optimizer'] == optimizer_str) & (df['transforms'] == transforms_str) & (df['neuron'] == str(neuron_viz)) & (df['recep_field'] == str(recep_field))]
	if len(df_sel) == 0:
		print('deepviz image not found for %s, generating . . .'%targetid)
		#image_name = 'deepviz_'+str(targetid)+'_'+objective+'_'+str(time.time())+'.jpg'
		image_name = str(targetid)+'_'+objective_str+'_'+str(time.time())+'.jpg'
		#gen_visualization(model,image_name,objective,parametrizer,optimizer,transforms,image_size,neuron,params) 
		full_image_path = viz_folder+'/featviz/'+image_name
		if parametrizer is None:
			parametrizer = lambda: param.image(image_size)
		print('generating featviz with objective: %s'%str(objective_str))
		if not recep_field:
			_ = render.render_vis(circuit, objective, parametrizer, optimizer, transforms=transforms,save_image=True,image_name=full_image_path, show_inline=True)
		else:
			featviz_in_recep_field(circuit,layer_name,within_id, input_dim = (3,224,224),image_name=full_image_path,device=device)
		with open(file_path, 'a') as csv:
			csv.write(','.join([image_name,str(targetid),objective_str,param_str,optimizer_str,transforms_str,str(neuron_viz),str(recep_field)])+'\n')
	else:
		print('found pre-generated image')
		image_name = df_sel.iloc[0]['image_name']  

	return image_name

	###feeding edge vizualizations, this is annoying, feed input node instead
	# df_sel = df.loc[(df['targetid'] == str(edgeid)) & (df['objective'] == objective) & (df['parametrizer'] == param_str) & (df['optimizer'] == optimizer_str) & (df['transforms'] == transforms_str)]
	# if len(df_sel) == 0:
	#     if 'r' in edgeid or 'g' in edgeid or 'b' in edgeid or 'gs' in edgeid:
	#         print('deepviz image not found for node input %s, generating . . .'%edgeid)
	#         #image_name = 'deepviz_'+str(targetid)+'_'+objective+'_'+str(time.time())+'.jpg'
	#         image_name = str(edgeid)+'_'+objective+'_'+str(time.time())+'.jpg'
	#         gen_visualization(model,image_name,objective,parametrizer,optimizer,transforms,image_size,params) 
	#         with open(file_path, 'a') as csv:
	#             csv.write(','.join([image_name,str(edgeid),objective,param_str,optimizer_str,transforms_str])+'\n')
	#     else:
	#         nodeid = edgeid.split('-')[0]
	#         print('no deepviz for edge %s for node inputs, looking for image for node %s'%(edgeid,nodeid))
	#         objective = gen_objective_str(nodeid,model,params)
	#         df_sel = df.loc[(df['targetid'] == str(nodeid)) & (df['objective'] == objective) & (df['parametrizer'] == param_str) & (df['optimizer'] == optimizer_str) & (df['transforms'] == transforms_str)]
	#         if len(df_sel) == 0:
	#             print('STILL no deepviz image found! generating now, but you should pregenerate these!')
	#             image_name = str(nodeid)+'_'+objective+'_'+str(time.time())+'.jpg'
	#             gen_visualization(model,image_name,objective,parametrizer,optimizer,transforms,image_size,params) 
	#         else:
	#             print('found pregenerated node image')
	#             image_name = df_sel.iloc[0]['image_name'] 
			
	# else:
	#     print('found pre-generated edge image')
	#     image_name = df_sel.iloc[0]['image_name']   
	

# def regen_visualization(model,targetid,neuron,params):
# 	model = set_across_model(model,'target_node',None)
# 	objective = gen_objective(targetid,model,params,neuron=neuron)
# 	parametrizer = params['deepviz_param']
# 	optimizer = params['deepviz_optim']
# 	transforms = params['deepviz_transforms']
# 	image_size = params['deepviz_image_size']

# 	param_str = object_2_str(parametrizer,"params['deepviz_param']=")
# 	optimizer_str = object_2_str(optimizer,"params['deepviz_optim']=")
# 	transforms_str = object_2_str(transforms,"params['deepviz_transforms']=")
# 	call('rm %s/visualizations/images/current_visualization*'%params['prepped_model_path'],shell=True )
# 	image_name = 'current_visualization'+str(time.time())+'.jpg'
# 	gen_visualization(model,image_name,objective,parametrizer,optimizer,transforms,image_size,neuron,params) 
# 	return image_name

# def combine_images(image_paths,heightwise=True):  #list of image paths
# 	images = [Image.open(x) for x in image_paths]
# 	widths, heights = zip(*(i.size for i in images))

# 	if heightwise:
# 		total_height = sum(heights)
# 		max_width = max(widths)

# 		new_im = Image.new('RGB', (max_width, total_height))

# 		y_offset = 0
# 		for im in images:
# 			new_im.paste(im, (0,y_offset))
# 			y_offset += im.size[1]

# 		return new_im
# 	else:
# 		total_width = sum(widths)
# 		max_height = max(heights)

# 		new_im = Image.new('RGB', (total_width, max_height))

# 		x_offset = 0
# 		for im in images:
# 			new_im.paste(im, (x_offset,0))
# 			x_offset += im.size[0]

# 		return new_im



def featviz_in_recep_field(model,layer,unit, input_dim = (3,224,224),image_name=None, margin = 10, device = 'cuda:0'):

	#dis_model = dissect_model(deepcopy(model), store_ranks = False, device=device)
	#dis_model = dis_model.to(device).eval()
	convert_relu_layers(model)
	#layer = layer.replace('.','_')

	#all_recep_field_params = receptive_field(model.features, input_dim)
	try:
		all_recep_field_params = receptive_field(model.to(device), input_dim)
	except:
		all_recep_field_params = receptive_field(model.features.to(device), input_dim)
	
	param_f = lambda: param.image(input_dim[1])
	obj  = objectives.neuron(layer,unit)
	viz_im = render.render_vis(model,  obj, param_f, show_inline=True)
	
	layer_ind = str(int(layer.replace('features.',''))+1)
	map_size = (all_recep_field_params[layer_ind]['output_shape'][2],all_recep_field_params[layer_ind]['output_shape'][3])
	print(map_size)
	if map_size[0]%2 == 0:
		target_position = (int(map_size[0]/2+1),int(map_size[1]/2+1))
	else:
		target_position = (int(floor(map_size[0]/2)),int(floor(map_size[1]/2)))
	print(target_position)
	recep_field = receptive_field_for_unit(all_recep_field_params, layer, target_position)
	print(recep_field)
	viz_im = viz_im[0][:,int(recep_field[0][0])-margin:int(recep_field[0][1])+margin,int(recep_field[1][0])-margin:int(recep_field[1][1])+margin,:]
	#viz_im = viz_im[0][:,:,:,:]
	image = (viz_im * 255).astype(np.uint8)
	if len(image.shape) == 4:
		image = np.concatenate(image, axis=1)
	image = Image.fromarray(image)

	if image_name is None:
		return image
	else:
		image.save(image_name)
	
