import numpy as np
import pickle
import scipy as sp
import scipy.sparse as spp

def load_sparse_matrix(filename):
	t = 0
	f = open(filename, 'r')
	f.readline()

	user_list = []
	item_list = []

	for line in f:
		word = line.split(',')
		user_list.append(int(word[0]))
		item_list.append(int(word[1]))

		t += 1
		if t % 50000 == 0:
			print('.', end = '')
			if t % 2000000 == 0:
				print('%d m lines' %(t // 1000000))

	users = list(set(user_list))
	items = list(set(item_list))
	user2id = dict(zip(users, range(len(users))))
	item2id = dict(zip(items, range(len(items))))

	data_list = list(np.ones(t))
	user_id_list = [user2id[u] for u in user_list]
	item_id_list = [item2id[i] for i in item_list]

	result = spp.csr_matrix((data_list, (user_id_list, item_id_list)))
	return result

def extract_users(num_users, sparse_matrix):
	item_count = sparse_matrix.getnnz(axis = 1)
	user_sort = sorted(enumerate(item_count), key = lambda x: x[1], reverse = True)[:num_users]
	user_indices = [x[0] for x in user_sort]

	return spp.vstack([sparse_matrix.getrow(i) for i in user_indices])

def extract_items(num_items, sparse_matrix):
	user_count = sparse_matrix.getnnz(axis = 0)
	item_sort = sorted(enumerate(user_count), key = lambda x: x[1], reverse = True)[:num_items]
	item_indics = [x[0] for x in item_sort]

	return spp.hstack([sparse_matrix.getcol(j) for j in item_indics])

def get_reduced_matrix(num_users, num_items, filename):
	data = load_sparse_matrix(filename)
	data1 = extract_users(3 * num_users, data)
	data2 = extract_items(num_items, data1)
	data3 = extract_users(num_users, data2)
	return data3.toarray()

reduced_matrix = get_reduced_matrix(num_users = 1000, num_items = 1000, filename = 'MovieLens/ratings.csv')
print(reduced_matrix[1][2])
# np.save('ml_1000user_1000item', reduced_matrix)
# X = np.load('ml_1000user_1000item.npy')
# print(X.shape)