Actual source code: embedgsvd.c

slepc-main 2025-01-19
Report Typos and Errors
  1: /*
  2:    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
  3:    SLEPc - Scalable Library for Eigenvalue Problem Computations
  4:    Copyright (c) 2002-, Universitat Politecnica de Valencia, Spain

  6:    This file is part of SLEPc.
  7:    SLEPc is distributed under a 2-clause BSD license (see LICENSE).
  8:    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
  9: */

 11: static char help[] = "Computes a GSVD associated with a complex network analysis problem.\n"
 12:   "The command line options are:\n"
 13:   "  -file <filename>, text file in TSV format containing undirected graph.\n"
 14:   "  -beta <beta>, optional decay parameter.\n\n";

 16: /*
 17:     Computes the Katz high-order proximity embedding of a graph via a partial GSVD.

 19:     [1] M. Ou et al. "Asymmetric transitivity preserving graph embedding" Proc. of
 20:         ACM SIGKDD 2016 - https://doi.org/10.1145/2939672.2939751
 21: */

 23: #include <slepcsvd.h>
 24: #include "network.h"

 26: PetscErrorCode SpectralRadius(Mat A,PetscReal *rho)
 27: {
 28:   EPS         eps;
 29:   PetscInt    nconv;
 30:   PetscScalar kr,ki;

 32:   PetscFunctionBeginUser;
 33:   PetscCall(EPSCreate(PETSC_COMM_WORLD,&eps));
 34:   PetscCall(EPSSetOperators(eps,A,NULL));
 35:   PetscCall(EPSSetProblemType(eps,EPS_NHEP));
 36:   PetscCall(EPSSetDimensions(eps,1,PETSC_DETERMINE,PETSC_DETERMINE));
 37:   PetscCall(EPSSetWhichEigenpairs(eps,EPS_LARGEST_MAGNITUDE));
 38:   PetscCall(EPSSetFromOptions(eps));
 39:   PetscCall(EPSSolve(eps));
 40:   PetscCall(EPSGetConverged(eps,&nconv));
 41:   *rho = 0.0;
 42:   if (nconv) {
 43:     PetscCall(EPSGetEigenvalue(eps,0,&kr,&ki));
 44:     *rho = SlepcAbsEigenvalue(kr,ki);
 45:   }
 46:   PetscCall(EPSDestroy(&eps));
 47:   PetscFunctionReturn(PETSC_SUCCESS);
 48: }

 50: int main(int argc,char **argv)
 51: {
 52:   Graph          graph;           /* directed graph */
 53:   Mat            A,B;             /* matrices */
 54:   SVD            svd;             /* singular value problem solver context */
 55:   PetscReal      beta=0.0,rho=0.0;
 56:   char           filename[PETSC_MAX_PATH_LEN];
 57:   PetscBool      flg,terse;

 59:   PetscFunctionBeginUser;
 60:   PetscCall(SlepcInitialize(&argc,&argv,NULL,help));

 62:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
 63:         Load graph and build adjacency matrix
 64:      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */

 66:   PetscCall(PetscOptionsGetString(NULL,NULL,"-graph",filename,sizeof(filename),&flg));
 67:   PetscCheck(flg,PETSC_COMM_WORLD,PETSC_ERR_USER_INPUT,"Must use -graph <name> to indicate a file containing the graph in TSV format");

 69:   PetscCall(GraphCreate(PETSC_COMM_WORLD,&graph));
 70:   PetscCall(GraphPreload(graph,filename));
 71:   PetscCheck(graph->type==GRAPH_DIRECTED,PETSC_COMM_SELF,PETSC_ERR_FILE_UNEXPECTED,"A directed graph is required");
 72:   PetscCall(PetscPrintf(PETSC_COMM_WORLD,"\nGraph with %" PetscInt_FMT " vertices and %" PetscInt_FMT " edges.\n\n",graph->nvertices,graph->nedges));
 73:   if (graph->weight!=GRAPH_WEIGHT_UNWEIGHTED) PetscCall(PetscPrintf(PETSC_COMM_WORLD,"WARNING: ignoring weights in input graph\n"));

 75:   PetscCall(GraphPreallocate(graph,filename));
 76:   PetscCall(GraphLoadUnweighted(graph,filename));

 78:   PetscCall(MatViewFromOptions(graph->adjacency,NULL,"-adjacency"));

 80:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
 81:       Build high-order proximity matrices (Katz): A=I-beta*G, B=beta*G
 82:      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */

 84:   PetscCall(PetscOptionsGetReal(NULL,NULL,"-beta",&beta,NULL));
 85:   if (beta==0.0) {  /* compute decay parameter beta */
 86:     PetscCall(PetscPrintf(PETSC_COMM_WORLD,"Computing spectral radius...\n"));
 87:     PetscCall(SpectralRadius(graph->adjacency,&rho));
 88:     if (rho) {
 89:       beta = 0.8*rho;
 90:       PetscCall(PetscPrintf(PETSC_COMM_WORLD,"Done, rho=%g, setting decay parameter beta=%g\n\n",(double)rho,(double)beta));
 91:     } else {
 92:       beta = 0.5;
 93:       PetscCall(PetscPrintf(PETSC_COMM_WORLD,"Failed, using default decay parameter beta=%g\n\n",(double)beta));
 94:     }
 95:   }
 96:   PetscCall(MatDuplicate(graph->adjacency,MAT_COPY_VALUES,&A));
 97:   PetscCall(MatScale(A,-beta));
 98:   PetscCall(MatShift(A,1.0));
 99:   PetscCall(MatDuplicate(graph->adjacency,MAT_COPY_VALUES,&B));
100:   PetscCall(MatScale(B,beta));

102:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
103:            Create the singular value solver and set various options
104:      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */

106:   PetscCall(SVDCreate(PETSC_COMM_WORLD,&svd));
107:   PetscCall(SVDSetOperators(svd,A,B));
108:   PetscCall(SVDSetProblemType(svd,SVD_GENERALIZED));
109:   PetscCall(SVDSetType(svd,SVDTRLANCZOS));
110:   PetscCall(SVDSetFromOptions(svd));

112:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
113:                       Solve the problem and print solution
114:      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */

116:   PetscCall(SVDSolve(svd));

118:   /* show detailed info unless -terse option is given by user */
119:   PetscCall(PetscOptionsHasName(NULL,NULL,"-terse",&terse));
120:   if (terse) PetscCall(SVDErrorView(svd,SVD_ERROR_NORM,NULL));
121:   else {
122:     PetscCall(PetscViewerPushFormat(PETSC_VIEWER_STDOUT_WORLD,PETSC_VIEWER_ASCII_INFO_DETAIL));
123:     PetscCall(SVDConvergedReasonView(svd,PETSC_VIEWER_STDOUT_WORLD));
124:     PetscCall(SVDErrorView(svd,SVD_ERROR_NORM,PETSC_VIEWER_STDOUT_WORLD));
125:     PetscCall(PetscViewerPopFormat(PETSC_VIEWER_STDOUT_WORLD));
126:   }
127:   PetscCall(SVDDestroy(&svd));
128:   PetscCall(GraphDestroy(&graph));
129:   PetscCall(MatDestroy(&A));
130:   PetscCall(MatDestroy(&B));
131:   PetscCall(SlepcFinalize());
132:   return 0;
133: }
134: /*TEST

136:    build:
137:       depends: network.o

139:    test:
140:       args: -graph ${SLEPC_DIR}/share/slepc/datafiles/graphs/out.moreno_taro_taro -svd_nsv 4 -terse
141:       filter: sed -e 's/4.38031/4.38032/' | sed -e 's/4.38033/4.38032/' | sed -e 's/3.75089/3.7509/' | sed -e 's/3.00071/3.00072/'

143: TEST*/