function main(network_type,view, networks_idx, test_percent, valid_percent, Total_methods, iters1, iters2)
networks = {'amazon-product', 'export', 'aidorganizations_issues', 'congressmen_topics_US', 'drug_target_GPCR_2009', 'drug_target_HQ_2014', 'drug_target_enzyme_2009', 'drug_target_ionchannel_2009', 'industries_eductionfields_IPUMS', 'occupations_tasks_ONET', 'tfs_genes_regulation_ecoli', 'users_movies_movielens100k', 'drug_target_moesm4_esm'};

networks = networks(networks_idx);
% Total_methods = {'NSA_batch', 'JCF', 'SSCF'};
merge_funcs = {'sum', 'normalization'};
CF_merge_funcs = {'normalization'};
measures = {'p','aup','aupr','auc','pbcorr', 'p_k', 'rec_k', 'ndcg_k'};
CHA_measures = {};
for i = 1:length(measures)
    CHA_measures = [CHA_measures ['CHA_' measures{i}] ];
end
measures = [measures CHA_measures];
allmeasures = {};
topks = [1, 5, 10, 20, 30, 40, 50];
for m2 = 1:length(measures)
    if endsWith(measures{m2}, '_k')
        for k = 1:length(topks)
            allmeasures = [allmeasures [measures{m2}(1:end-2) '_at_'  num2str(topks(k))]];
        end
    else
        allmeasures = [allmeasures measures{m2}];
    end
end
denominators = {'sum_of_degree', 'sum_of_nlcl', 'union_of_neighbors'};
alphas = linspace(0, 1, 11);
exps = [1, 2];
processings = {'nothing'};
parameters = {'denominator', 'exp', 'processing', 'alpha', 'merge_func'};
CF_parameters = {'alpha'};
index_dic = containers.Map( ...
    {'NSA_val', 'NSA_batch', 'NSA3.1_val', 'NSA3.1_batch'}, ...
    {'CH3_L2', 'CH3_L2', 'CH3v5a_L2', 'CH3v5a_L2'});

for j = 3:iters1
    for i = 1:length(networks)
        load(['./' network_type '_dataset/' view '_based/' num2str(test_percent) '/matrices/' networks{i} '.mat'], 'xx')
        load(['./' network_type '_dataset/' view '_based/' num2str(test_percent) '/sparsified_matrices/' networks{i} '_train.mat'], 'matrices')
        x = xx{1};
        fprintf('%d/%d: %s (%dx%d) ... ', i, length(networks), networks{i}, size(x,1), size(x,2));
        for tm = 1:length(Total_methods)
            x = xx{j};
            adj = matrices{j};
            if (Total_methods{tm} == "SSCF") || (Total_methods{tm} == "JCF")
                root = [network_type '_result/' view '_based/' num2str(test_percent) '/' Total_methods{tm} '/' networks{i} '/' num2str(j) '/'];
                CF_batch(root, adj, x, Total_methods{tm}, CF_merge_funcs, alphas, topks, measures, CF_parameters);
            elseif (Total_methods{tm} == "SSCF_val") || (Total_methods{tm} == "JCF_val")
                
                load(['./' network_type '_dataset/' view '_based/' num2str(test_percent) '/val_matrices/' num2str(valid_percent) '/ori/' networks{i} '.mat'], 'ori_matrices')
                load(['./' network_type '_dataset/' view '_based/' num2str(test_percent) '/val_matrices/' num2str(valid_percent) '/train/' networks{i} '.mat'], 'train_matrices')
                root = [network_type '_result/' view '_based/' num2str(test_percent) '/' Total_methods{tm} '/' num2str(valid_percent) '/' networks{i} '/' num2str(j) '/val/'];
                for k = 1:iters2
                    CF_batch([root num2str(k) '/'], train_matrices{j}{k}, ori_matrices{j}{k}, Total_methods{tm}, CF_merge_funcs, alphas, topks, measures, CF_parameters);
                end
                root1 = [network_type '_result/' view '_based/' num2str(test_percent) '/' Total_methods{tm}(1:end-4) '/' networks{i} '/' num2str(j) '/'];
                root2 = [network_type '_result/' view '_based/' num2str(test_percent) '/' Total_methods{tm} '/' num2str(valid_percent) '/' networks{i} '/' num2str(j) '/'];
                for mm = 1:length(allmeasures)
                    tunedparameter = find_tunedparameter(CF_parameters, iters2, root, allmeasures{mm});
                    target = CF_one(root1, adj, x, Total_methods{tm}, tunedparameter{1}, topks, measures, allmeasures{mm});
                    save([root2 allmeasures{mm} '.mat'], 'target', 'tunedparameter', 'CF_parameters')
                end
                
            elseif (Total_methods{tm} == "NSA_batch") || (Total_methods{tm} == "NSA3.1_batch")
                root = [network_type '_result/' view '_based/' num2str(test_percent) '/' Total_methods{tm} '/' networks{i} '/' num2str(j) '/'];
                NSA_batch(root, adj, x, denominators, exps, processings, merge_funcs, alphas, topks, measures, parameters, index_dic(Total_methods{tm}));
            elseif (Total_methods{tm} == "NSA_val") || (Total_methods{tm} == "NSA3.1_val")
                
                load(['./' network_type '_dataset/' view '_based/' num2str(test_percent) '/val_matrices/' num2str(valid_percent) '/ori/' networks{i} '.mat'], 'ori_matrices')
                load(['./' network_type '_dataset/' view '_based/' num2str(test_percent) '/val_matrices/' num2str(valid_percent) '/train/' networks{i} '.mat'], 'train_matrices')
                root = [network_type '_result/' view '_based/' num2str(test_percent) '/' Total_methods{tm} '/' num2str(valid_percent) '/' networks{i} '/' num2str(j) '/val/'];
                for k = 1:iters2
                    NSA_batch([root num2str(k) '/'], train_matrices{j}{k}, ori_matrices{j}{k}, denominators, exps, processings, merge_funcs, alphas, topks, measures, parameters, index_dic(Total_methods{tm}));
                end
            end
        end
    end
end


function CF_batch(root, adj, x, method, merge_funcs, alphas, topks, measures, parameters)
[item_sim, user_sim] = compute_similarity(method, adj);
for merge_func = 1:length(merge_funcs)
    [item_one_mode, user_one_mode] = merge(merge_funcs{merge_func}, adj, item_sim, user_sim);
    for alpha = 1:length(alphas)
        out = [root 'alpha' num2str(alphas(alpha)) '.mat'];
        if exist(out, 'file')
            continue;
        end
        half_half = alphas(alpha) * item_one_mode + (1-alphas(alpha))*user_one_mode;
        checkdir(root);
        value = evaluate(half_half, adj, x, topks, measures, out);
        this_parameter = containers.Map(parameters, {alphas(alpha)});
        savebest(root, this_parameter, measures, topks, value)
    end
end

function target = CF_one(root, adj, x, method, alpha, topks, measures, target_measure)
out = [root 'alpha' num2str(alpha) '.mat'];
if exist(out, "file")
    if contains(target_measure, '_at_')
        splitStr = split(target_measure, '_at_');
        load(out, [splitStr{1} '_k'])
        a = int32(str2double(splitStr{2}));
        eval(sprintf('target = %s(%s);', [splitStr{1} '_k'], num2str(find(topks == a))));
    else
        load(out, target_measure)
        eval(sprintf('target = %s;', target_measure));
    end
    return
end
[item_sim, user_sim] = compute_similarity(method, adj);
[item_one_mode, user_one_mode] = merge('normalization', adj, item_sim, user_sim);

half_half = alpha * item_one_mode + (1-alpha)*user_one_mode;
checkdir(root);
value = evaluate(half_half, adj, x, topks, measures, out);
target = value(target_measure);



function NSA_batch(root, adj, x, denominators, exps, processings, merge_funcs, alphas, topks, measures, parameters, index)
[item_C1, info] = CHA_linkpred_bipartite(adj', {index}, 0, 100);
[user_C1, info] = CHA_linkpred_bipartite(adj, {index}, 0, 100);
item_sim = item_C1;
user_sim = user_C1;
item_degree = sum(adj, 1);
user_degree = sum(adj, 2);
for denominator = 1:length(denominators)
    [item_dom, user_dom] = compute_NSA_dom(denominators{denominator}, adj, item_degree, user_degree);
    for exp = 1:length(exps)
        item_scaled_sim = item_sim ./ (item_dom .^ exps(exp));
        user_scaled_sim = user_sim ./ (user_dom .^ exps(exp));
        for processing = 1:length(processings)
            item_scaled_sim1 = add_op_func(item_scaled_sim, processings{processing});
            user_scaled_sim1 = add_op_func(user_scaled_sim, processings{processing});
            for merge_func = 1:length(merge_funcs)
                [item_one_mode, user_one_mode] = merge(merge_funcs{merge_func}, adj, item_scaled_sim1, user_scaled_sim1);
                for alpha = 1:length(alphas)
                    out = [root denominators{denominator} '_exp' num2str(exps(exp)) '_' processings{processing} '_alpha' num2str(alphas(alpha)) '_' merge_funcs{merge_func} '.mat'];
                    if exist(out, 'file')
                        continue;
                    end
                    half_half = alphas(alpha) * item_one_mode + (1-alphas(alpha))*user_one_mode;
                    checkdir(root);
                    value = evaluate(half_half, adj, x, topks, measures, out);
                    this_parameter = containers.Map(parameters, {denominators{denominator}, exps(exp), processings{processing}, alphas(alpha), merge_funcs{merge_func}});
                    savebest(root, this_parameter, measures, topks, value)
                end
            end
        end
    end
end

function target = NSA_one(root, adj, x, denominator, exp, processing, merge_func, alpha, topks, measures, target_measure, index)
out = [root denominator '_exp' num2str(exp) '_' processing '_alpha' num2str(alpha) '_' merge_func '.mat'];
if exist(out, "file")
    if contains(target_measure, '_at_')
        splitStr = split(target_measure, '_at_');
        load(out, [splitStr{1} '_k'])
        a = int32(str2double(splitStr{2}));
        eval(sprintf('target = %s(%s);', [splitStr{1} '_k'], num2str(find(topks == a))));
    else
        load(out, target_measure)
        eval(sprintf('target = %s;', target_measure));
    end
    return
end

[item_C1, info] = CHA_linkpred_bipartite(adj', {index}, 0, 100);
[user_C1, info] = CHA_linkpred_bipartite(adj, {index}, 0, 100);
item_sim = item_C1;
user_sim = user_C1;
item_degree = sum(adj, 1);
user_degree = sum(adj, 2);
[item_dom, user_dom] = compute_NSA_dom(denominator, adj, item_degree, user_degree);
item_scaled_sim = item_sim ./ (item_dom .^ exp);
user_scaled_sim = user_sim ./ (user_dom .^ exp);
item_scaled_sim1 = add_op_func(item_scaled_sim, processing);
user_scaled_sim1 = add_op_func(user_scaled_sim, processing);
[item_one_mode, user_one_mode] = merge(merge_func, adj, item_scaled_sim1, user_scaled_sim1);
half_half = alpha * item_one_mode + (1-alpha)*user_one_mode;
out = [root denominator '_exp' num2str(exp) '_' processing '_alpha' num2str(alpha) '_' merge_func '.mat'];
checkdir(root);
value = evaluate(half_half, adj, x, topks, measures, out);
target = value(target_measure);


function [item_sim, user_sim] = compute_similarity(method, adj)
if (method == "SSCF") || (method == "SSCF_val")
    [item_sim, user_sim] = SaplingSimilarity(adj);
elseif (method == "JCF") || (method == "JCF_val")
    [item_sim, user_sim] = JaccardSimilarity(adj);
end

function [item_sap, user_sap] = SaplingSimilarity(M)
N = size(M, 1);
k = sum(M, 1);
CO = M' * M;
B1 = (1-(CO.*(1-CO./k)+(k-CO')'.*(1-(k-CO')'./(N-k)))'./(k.*(1-k./N)))'.*sign(((CO.*N./k)'./k)' - 1);
B1(isnan(B1)) = 0;

N = size(M, 2);
k = sum(M, 2);
CO = M * M';
B2 = (1-(CO.*(1-CO./k)+(k-CO')'.*(1-(k-CO')'./(N-k)))'./(k.*(1-k./N)))'.*sign(((CO.*N./k)'./k)' - 1);
B2(isnan(B2)) = 0;

item_sap = B1;
user_sap = B2;

function [item_jac, user_jac] = JaccardSimilarity(adj)
item_degree = sum(adj, 1);
user_degree = sum(adj, 2);

item_I = adj' * adj;
item_U = item_degree' + item_degree - item_I;
item_jac = item_I ./ item_U;

user_I = adj * adj';
user_U = user_degree + user_degree' - user_I;
user_jac = user_I ./ user_U;

function [item_dom, user_dom] = compute_NSA_dom(denominator, adj, item_degree, user_degree)
if denominator == "sum_of_degree"
    item_dom = item_degree' + item_degree;
    user_dom = user_degree + user_degree';
elseif denominator == "sum_of_nlcl"
    item_nlcl = item_degree' + item_degree - adj' * adj * 2;
    user_nlcl = user_degree + user_degree' - adj * adj' * 2;
    item_dom = item_nlcl+1;
    user_dom = user_nlcl+1;
elseif denominator == "union_of_neighbors"
    item_U = item_degree' + item_degree - adj' * adj;
    user_U = user_degree + user_degree' - adj * adj';
    item_dom = item_U;
    user_dom = user_U;
end


function [item_sap_one_mode, user_sap_one_mode] = merge(merge_func, adj, item_sap, user_sap)
if merge_func == "sum"
    item_sap_one_mode = (item_sap * adj')';
    user_sap_one_mode = user_sap * adj; 
elseif merge_func == "normalization"
    item_degree = repmat(sum(abs(item_sap), 2), 1,size(adj', 2));
    temp = ones(size(item_degree));
    temp(item_degree == 0) = 0;
    item_degree(item_degree == 0) = 1;
    item_sap_one_mode = ((item_sap * adj') ./ item_degree .* temp)';

    user_degree = repmat(sum(abs(user_sap), 2), 1, size(adj, 2));
    temp = ones(size(user_degree));
    temp(user_degree == 0) = 0;
    user_degree(user_degree == 0) = 1;
    user_sap_one_mode = (user_sap * adj) ./ user_degree .* temp; 
elseif merge_func == "sub"
    item_degree = repmat(sum(abs(item_sap), 2), 1,size(adj', 2));
    temp = ones(size(item_degree));
    temp(item_degree == 0) = 0;
    item_degree(item_degree == 0) = 1;
    item_sap_one_mode = 1+((item_sap * (adj'-1)) ./ item_degree .* temp)';

    user_degree = repmat(sum(abs(user_sap), 2), 1, size(adj, 2));
    temp = ones(size(user_degree));
    temp(user_degree == 0) = 0;
    user_degree(user_degree == 0) = 1;
    user_sap_one_mode = 1+((user_sap * (adj-1)) ./ user_degree .* temp);
elseif merge_func == "mean"
    item_mono_adj = adj' * adj;
    item_mono_adj(item_mono_adj>0) = 1;
    item_degree = item_mono_adj * adj';
    temp = ones(size(item_degree));
    temp(item_degree == 0) = 0;
    item_degree(item_degree == 0) = 1;
    item_sap_one_mode = ((item_sap * adj') ./ item_degree .* temp)';
    
    user_mono_adj = adj * adj';
    user_mono_adj(user_mono_adj>0) = 1;
    user_degree = user_mono_adj * adj;
    temp = ones(size(user_degree));
    temp(user_degree == 0) = 0;
    user_degree(user_degree == 0) = 1;
    user_sap_one_mode = (user_sap * adj) ./ user_degree .* temp;
end

function half_half = add_op_func(half_half, method)
if strcmp(method, 'nothing')
    return
end
half_half = abs(half_half - min(half_half(:)) - max(half_half(:)));
G = graph(half_half);
half_half = distances(G);
if strcmp(method, 'euclidean')
    half_half = squareform(pdist(half_half, method));
    half_half = abs(half_half - min(half_half(:)) - max(half_half()));
elseif strcmp(method, 'pearson')
    half_half = (corrcoef(half_half) + 1) / 2;
end

function value = evaluate(half_half, adj, x, topks, measures, out)
[CHA_p_k, CHA_rec_k, CHA_ndcg_k] = prediction_evaluation_ks(half_half, adj, topks);
[CHA_p, CHA_aup, CHA_aupr, CHA_auc, CHA_pbcorr] = prediction_evaluation_20(full(half_half(:)), full(adj(:)));

half_half(adj==1)=0;
new_labels = x;
new_labels(adj==1) = 0;
[p_k, rec_k, ndcg_k] = prediction_evaluation_ks(half_half, new_labels, topks);

[row, col, value] = find(half_half);
labels = x(sub2ind(size(x),row,col));
[p, aup, aupr, auc, pbcorr] = prediction_evaluation_20(value, labels);
save(out, measures{:});
value = containers.Map();
for m2 = 1:length(measures)
    if endsWith(measures{m2}, '_k')
        for k = 1:length(topks)
            eval(sprintf('value(''%s_at_%s'') = %s(%s);', measures{m2}(1:end-2), num2str(topks(k)), measures{m2}, num2str(k)));
        end
    else
        eval(sprintf('value(''%s'') = %s;', measures{m2}, measures{m2}));
    end
end

function [p_k, rec_k, ndcg_k] = prediction_evaluation_ks(half_half, new_labels, topks)
p_k = [];
rec_k = [];
ndcg_k = [];
for k = 1:length(topks)
    [p_k(k), rec_k(k), ndcg_k(k)] = prediction_evaluation_k(half_half, new_labels, topks(k));
end

function checkdir(out)
if ~exist(out, 'dir')
    mkdir(out);
end

function savebest(root, this_parameter, measures, topks, value)
if ~exist([root 'best.mat'], 'file')
    bestvalue = value;
    bestparameter = containers.Map();
    for m2 = 1:length(measures)
        if endsWith(measures{m2}, '_k')
            for k = 1:length(topks)
                eval(sprintf('bestparameter(''%s_at_%s'') = this_parameter;', measures{m2}(1:end-2), num2str(topks(k))));
            end
        else
            eval(sprintf('bestparameter(''%s'') = this_parameter;', measures{m2})); 
        end
    end
    save([root 'best.mat'], 'bestparameter', 'bestvalue');
else
    load([root 'best.mat'], 'bestparameter', 'bestvalue');
    allKeys = keys(value);
    for key = 1:length(allKeys)
        if value(allKeys{key}) > bestvalue(allKeys{key})
            bestvalue(allKeys{key}) = value(allKeys{key});
            bestparameter(allKeys{key}) = this_parameter;
        end
    end
    save([root 'best.mat'], 'bestparameter', 'bestvalue');
end

function tunedparameter = find_tunedparameter(parameters, iters2, root, target_measure)
myparameter = cell(length(parameters), iters2);
for k = 1:iters2
    path = [root num2str(k) '/best.mat'];
    load(path, 'bestparameter');
    thisparameter = bestparameter(target_measure);
    for p = 1:length(parameters)
        myparameter{p, k} = thisparameter(parameters{p});
    end
end
tunedparameter = cell(length(parameters), 1);
for p = 1:length(parameters)
    row = myparameter(p, :);
    if (strcmp(parameters{p}, 'exp')) || (strcmp(parameters{p}, 'alpha'))
        tunedparameter{p} = mode(cell2mat(row));
    else
        [uniqueStrs, ~, idx] = unique(row);
        counts = histcounts(idx, 1:numel(uniqueStrs)+1);
        [~, maxIdx] = max(counts);
        tunedparameter{p} = uniqueStrs{maxIdx};
    end
end







            
            