/*
experiment: speedup for Local Search for k-Robust Median.
*/

#include "../lib/head.h"
#include "../lib/tools.h"
#include "../lib/function.h"
#include "../lib/kmeans.h"
#include "../lib/coreset.h"

double Mean(vector<double> a){
	double sum = 0;
	for(double x : a) sum += x;
	return sum / (int)a.size();
}

double Max(vector<double> a){
	double ret = 0;
	for(double x : a) ret = max(ret, x);
	return ret;
}

double Min(vector<double> a){
	double ret = 1;
	for(double x : a) ret = min(ret, x);
	return ret;
}

double Var(vector<double> a){
	double ret = 0, ave = Mean(a);
	for(double x : a) ret += (x - ave) * (x - ave);
	return ret / (int)a.size();
}

int main(int argc, char **argv){
	if(argc < 8){
		printf("7 arguments are required (data_name, data_size, data_dim, num_of_centers, num_of_outliers, coreset_size, threshold).");
		return 0;
	}
	FILE *FIN = fopen(argv[1],"r");
	//N : number of datapoints;
	//D : dimension;
	//k : number of clusters;
	//m : number of outliers;
	//mm : size of coreset minus number of outliers, that is, we construct a coreset with size m + mm;
	//thr : used to determine how the rings and groups
	int N = atoi(argv[2]), D = atoi(argv[3]), k =atoi(argv[4]), m = atoi(argv[5]), mm = atoi(argv[6]);
	double thr = atof(argv[7]);
	dataset X;
	cerr<<"-----------Reading Data-----------"<<endl;
	for(int i = 1; i <= N; i++){
		datapoint x; x.resize(D + 1);
		x[0] = 1;
		for(int j = 1; j <= D; j++)
			fscanf(FIN, "%lf", &x[j]);
		X.push_back(x);
	}
	cerr<<"-----------Reading Over-----------"<<endl;
	cerr<<"-----------Preparing for Local Search-----------" << endl;
	dataset C_init;
	dataset candi;
	for(int i = 1; i <= 500; i++) candi.push_back(X[(int)(randm() * candi.size())]);
	cerr<<"-----------Preparing End-----------"<<endl;
	int T = 10;
	dataset OurCoreset, UniCoreset;
	double ave_our_construct_ti = 0, ave_our_lloyd_ti = 0;
	double ave_uni_construct_ti = 0, ave_uni_lloyd_ti = 0;
	double ave_ti = 0;
	vector<double> ores, ures, res;
	for(int t = 1; t <= T; t++){
		C_init.clear();
		for(int i = 1; i <= k; i++) C_init.push_back(candi[(int)(randm() * candi.size())]);
		double ti = clock();
		dataset C = one_swap_local_search(X, C_init, candi, m, 0.01, 100);
		double Xti = (clock() - ti)/(double)CLOCKS_PER_SEC;

		double xcost = rcost(X, C, m, 1);
		//cerr << Xti << " " << xcost << endl;
		ave_ti += Xti;
		ti = clock();
		OurCoreset = Our_Coreset(X, Approx_KRMeans(X, k, m, 1, -1, 1), m, mm, 1, thr);
		double our_construct_ti = (clock() - ti)/ (double)CLOCKS_PER_SEC;
		
		ti = clock();
		C = one_swap_local_search(OurCoreset, C_init, candi, m, 0.01, 100);
		double our_lloyd_ti = (clock() - ti)/ (double)CLOCKS_PER_SEC;

		double ocost = rcost(X, C, m, 1);

		ave_our_construct_ti += our_construct_ti;
		ave_our_lloyd_ti += our_lloyd_ti;

		ores.push_back(ocost);
		res.push_back(xcost);
		cerr <<"Round " << t << ". running on original dataset : " << Xti <<"; cost : " << xcost<< endl;
		cerr <<"Round " << t << ". ours, construction time : " << our_construct_ti << "; local search time : " << our_lloyd_ti << "; cost : " << ocost << endl;
	}
	cerr << "Testing end." << endl;
	cerr << "local search on original dataset , time : " << ave_ti / T << "; cost : " << Mean(res) << endl;
	cerr << "ours, total time : " << ave_our_construct_ti / T + ave_our_lloyd_ti / T << "; construction time : " << ave_our_construct_ti / T << "; local search time : " << ave_our_lloyd_ti / T << "; cost : " << Mean(ores) << endl;
	return 0;
}