From c44923a13d68b5620e913f5e64fff2a83fe47dea Mon Sep 17 00:00:00 2001 From: Theodore Vasiloudis Date: Tue, 29 Oct 2024 15:31:24 -0700 Subject: [PATCH 1/2] Allow using multiple metrics when using early stop. (#1079) *Issue #, if available:* *Description of changes:* * Previously we had an assertion that did not allow a dict with two metrics to be passed to the early stop check function. * This change removes that and adds a case in the e2e tests that does early stop but has more than one eval metric. Example error we used to get: ``` Traceback (most recent call last): [rank77]: File "/graphstorm/python/graphstorm/run/gsgnn_lp/gsgnn_lp.py", line 180, in [rank77]: main(gs_args) [rank77]: File "/graphstorm/python/graphstorm/run/gsgnn_lp/gsgnn_lp.py", line 136, in main [rank77]: trainer.fit(train_loader=dataloader, val_loader=val_dataloader, [rank77]: File "/graphstorm/python/graphstorm/trainer/lp_trainer.py", line 247, in fit [rank77]: if self.evaluator.do_early_stop(val_score): [rank77]: File "/graphstorm/python/graphstorm/eval/evaluator.py", line 358, in do_early_stop [rank77]: assert len(val_score) == 1, \ [rank77]: AssertionError: validation score should be a single key value pair but got {'amri': 0.688815712928772, 'hit_at_10': 0.13315722346305847, 'mrr': 0.06222189590334892} ``` By submitting this pull request, I confirm that you can use, modify, copy, and redistribute this contribution, under the terms of your choice. --- python/graphstorm/eval/evaluator.py | 2 -- tests/end2end-tests/graphstorm-lp/mgpu_test.sh | 2 +- 2 files changed, 1 insertion(+), 3 deletions(-) diff --git a/python/graphstorm/eval/evaluator.py b/python/graphstorm/eval/evaluator.py index 96cf1113cb..875cd1f46d 100644 --- a/python/graphstorm/eval/evaluator.py +++ b/python/graphstorm/eval/evaluator.py @@ -355,8 +355,6 @@ def do_early_stop(self, val_score): if self._do_early_stop is False: return False - assert len(val_score) == 1, \ - f"validation score should be a single key value pair but got {val_score}" self._num_early_stop_calls += 1 # Not enough existing validation scores if self._num_early_stop_calls <= self._early_stop_burnin_rounds: diff --git a/tests/end2end-tests/graphstorm-lp/mgpu_test.sh b/tests/end2end-tests/graphstorm-lp/mgpu_test.sh index 439b7bf9ff..33338418e9 100644 --- a/tests/end2end-tests/graphstorm-lp/mgpu_test.sh +++ b/tests/end2end-tests/graphstorm-lp/mgpu_test.sh @@ -562,7 +562,7 @@ then fi echo "**************dataset: Movielens, RGCN layer 2, node feat: fixed HF BERT & sparse embed, BERT nodes: movie, inference: full-graph, negative_sampler: joint, exclude_training_targets: true, save model, early stop" -python3 -m graphstorm.run.launch --workspace $GS_HOME/training_scripts/gsgnn_lp --num-trainers $NUM_TRAINERS --num-servers 1 --num-samplers 0 --part-config /data/movielen_100k_lp_train_val_1p_4t/movie-lens-100k.json --ip-config ip_list.txt --ssh-port 2222 $GS_HOME/python/graphstorm/run/gsgnn_lp/gsgnn_lp.py --cf ml_lp.yaml --fanout '10,15' --num-layers 2 --use-mini-batch-infer false --use-node-embeddings true --exclude-training-targets True --reverse-edge-types-map user,rating,rating-rev,movie --save-model-path /data/gsgnn_lp_ml_dot/ --topk-model-to-save 3 --save-model-frequency 1000 --save-embed-path /data/gsgnn_lp_ml_dot/emb/ --use-early-stop True --early-stop-burnin-rounds 3 -e 30 --early-stop-rounds 2 --early-stop-strategy consecutive_increase --logging-file /tmp/exec.log +python3 -m graphstorm.run.launch --workspace $GS_HOME/training_scripts/gsgnn_lp --num-trainers $NUM_TRAINERS --num-servers 1 --num-samplers 0 --part-config /data/movielen_100k_lp_train_val_1p_4t/movie-lens-100k.json --ip-config ip_list.txt --ssh-port 2222 $GS_HOME/python/graphstorm/run/gsgnn_lp/gsgnn_lp.py --cf ml_lp.yaml --fanout '10,15' --num-layers 2 --use-mini-batch-infer false --use-node-embeddings true --exclude-training-targets True --reverse-edge-types-map user,rating,rating-rev,movie --save-model-path /data/gsgnn_lp_ml_dot/ --topk-model-to-save 3 --save-model-frequency 1000 --save-embed-path /data/gsgnn_lp_ml_dot/emb/ --use-early-stop True --early-stop-burnin-rounds 3 -e 30 --early-stop-rounds 2 --early-stop-strategy consecutive_increase --logging-file /tmp/exec.log --eval-metric mrr amri error_and_exit $? From ee16e74cf180932687092c9b4478d9a2fc8214f7 Mon Sep 17 00:00:00 2001 From: Theodore Vasiloudis Date: Wed, 30 Oct 2024 09:17:41 -0700 Subject: [PATCH 2/2] Add documentation for AMRI and other LP eval metrics (#1075) *Issue #, if available:* *Description of changes:* * Add rst docs for AMRI and other LP eval metrics By submitting this pull request, I confirm that you can use, modify, copy, and redistribute this contribution, under the terms of your choice. --- docs/source/advanced/link-prediction.rst | 77 ++++++++++++++++++- .../configuration-run.rst | 19 ++++- python/graphstorm/eval/evaluator.py | 2 + .../graphstorm/run/gsgnn_lp/lp_infer_gnn.py | 10 ++- python/graphstorm/run/gsgnn_lp/lp_infer_lm.py | 10 ++- .../end2end-tests/graphstorm-lp/mgpu_test.sh | 11 ++- 6 files changed, 117 insertions(+), 12 deletions(-) diff --git a/docs/source/advanced/link-prediction.rst b/docs/source/advanced/link-prediction.rst index 10d7450cb6..9f910c010e 100644 --- a/docs/source/advanced/link-prediction.rst +++ b/docs/source/advanced/link-prediction.rst @@ -269,7 +269,8 @@ In general, GraphStorm covers following cases: - **Case 2** ``num_train_hard_negatives`` is smaller than ``num_negative_edges``. GraphStorm will randomly sample ``num_train_hard_negatives`` hard negative nodes from the hard negative set and then randomly sample ``num_negative_edges - num_train_hard_negatives`` negative nodes. - **Case 3** GraphStorm supports cases when some edges do not have enough hard negatives provided by users. For example, the expected ``num_train_hard_negatives`` is 10, but an edge only have 5 hard negatives. In certain cases, GraphStorm will use all the hard negatives first and then randomly sample negative nodes to fulfill the requirement of ``num_train_hard_negatives``. Then GraphStorm will go back to **Case 1** or **Case 2**. -**Preparing graph data for hard negative sampling** +Preparing graph data for hard negative sampling +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ The gconstruct pipeline of GraphStorm provides support to load hard negative data from raw input. Hard destination negatives can be defined through ``edge_dst_hard_negative`` transformation. @@ -328,3 +329,77 @@ For example, the file storing hard negatives should look like the following: "src_100"| "dst_41"| "dst0;dst_2" GraphStorm will automatically translate the Raw Node IDs of hard negatives into Partition Node IDs in a DistDGL graph. + +.. _link-prediction-evaluation-metrics: + +Link Prediction Metrics +----------------------- + +GraphStorm supports several metrics for link prediction, to give a well-rounded +view of model performance. +In general, link prediction evaluation happens by constructing a set of negative +edges with one of the sampling methods described above, and including one positive +edge in this set of edges, which we will refer to as the `candidate set`. The model +assigns a score to each edge in the candidate set, and ideally the true edge is ranked +at the top position when edges are ranked by scores. + +We define the set of ranking scores as :math:`\mathcal{I}` and the number of candidate +edges as :math:`\mathcal{|I|}`. We refer to the ranking of a positive edge within the list +as :math:`r`. + +Mean Reciprocal Rank (MRR) +^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Mean Reciprocal Rank or MRR is a metric commonly used in link prediction evaluation +that represents the ability of the model to rank the correct edge among a list of +candidate edges. It is defined as: + +.. math:: + + \text{MRR} = \frac{1}{| \mathcal{I} |} \sum_{r \in \mathcal{I}}{\frac{1}{r}} + +where :math:`\mathcal{I}` is the set of candidate edges, and :math:`r` corresponds to the +ranking of the positive edge as determined by the score assigned to the model to +each edge in the candidate set. + +The ideal MRR is 1.0 meaning that the positive edges are ranked first in every +score list. Because a positive edge is always included in the ranking, it cannot +get the value of 0.0 so its range is in :math:`(0, 1]`. MRR values are influenced by +the size of the candidate lists, so it can only be used to compare the performance +when the number of negative edges per positive edge is the same. + +Hits@k +^^^^^^ + +The ``Hits@k`` metric measures the number of times the positive edge was ranked in the +top k positions by the model in the sorted score list: + +.. math:: + + \text{Hits@k} = \frac{| r \in \mathcal{I} | r \leq k |}{| \mathcal{I} |} + +This metric is easy to interpret but has the disadvantage that any position +beyond the top-k is not taken into account, so does not provide a holistic +view needed for cross-model comparison, and is also sensitive to the number +of negatives in the set. + + +Adjusted Mean Ranking Index (AMRI) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +AMRI was proposed in the paper +`On the Ambiguity of Rank-Based Evaluation of EA or LP Methods `_ +as a metric that allows cross-model comparison, by looking at the entire score list, but is not +sensitive to the chosen number of negative edges per positive edge. It is defined as: + + +.. math:: + + \text{AMRI} = 1 - \frac{\text{MR}-1}{\mathbb{E}[\text{MR}-1]} + +where :math:`\text{MR}` is the mean rank, and :math:`\mathbb{E}[\text{MR}-1]` is the expected mean rank, +which is used to adjust for chance. Its values will be in the :math:`[-1, 1]` range, where 1 corresponds +to optimal performance where each individual rank of the positive edge is 1. A value of 0 indicates +model performance similar to a model assigning random scores, or equal score +to every candidate. The value is negative if the model performs worse than the +all-equal-score model." diff --git a/docs/source/cli/model-training-inference/configuration-run.rst b/docs/source/cli/model-training-inference/configuration-run.rst index cc96bae17c..f363a8ac57 100644 --- a/docs/source/cli/model-training-inference/configuration-run.rst +++ b/docs/source/cli/model-training-inference/configuration-run.rst @@ -343,7 +343,24 @@ General Configurations - Default value: This parameter must be provided by user. .. _eval_metrics: -- **eval_metric**: Evaluation metric used during evaluation. The input can be a string specifying the evaluation metric to report or a list of strings specifying a list of evaluation metrics to report. The first evaluation metric is treated as the major metric and is used to choose the best trained model. The supported evaluation metrics of classification tasks include ``accuracy``, ``precision_recall``, ``roc_auc``, ``f1_score``, ``per_class_f1_score``, ``hit_at_k``. To be noted, ``hit_at_k`` only works with binary classification tasks. The ``k`` of ``hit_at_k`` can be any positive integer, for example ``hit_at_10`` or ``hit_at_100``. The term ``hit_at_k`` refers to the number of true positives among the top ``k`` predictions with the highest confidence scores. The supported evaluation metrics of regression tasks include ``rmse``, ``mse`` and ``mae``. The supported evaluation metrics of link prediction tasks include ``mrr`` and ``hit_at_k``. +- **eval_metric**: Evaluation metrics used during evaluation. The input can be a string specifying + the evaluation metric to report or a list of strings specifying a list of evaluation metrics to + report. The first evaluation metric in the list is treated as the primary metric and is used to + choose the best trained model and for early stopping. Each learning task supports different evaluation metrics: + + - The supported evaluation metrics of classification tasks include ``accuracy``, + ``precision_recall``, ``roc_auc``, ``f1_score``, ``per_class_f1_score``, ``hit_at_k``. Note that + ``hit_at_k`` only works with binary classification tasks. + + - The ``k`` of ``hit_at_k`` can be any positive integer, for example ``hit_at_10`` or + ``hit_at_100``. The term ``hit_at_k`` refers to the number of true positives among the top ``k`` + predictions with the highest confidence scores. + - The supported evaluation metrics of regression tasks include ``rmse``, ``mse`` and ``mae``. + - The supported evaluation metrics of link prediction tasks include ``mrr``, ``amri`` and + ``hit_at_k``. MRR refers to the Mean Reciprocal Rank with values between and 0 (worst) and 1 + (best), and AMRI refers the Adjusted Mean Rank Index, with values ranging from -1 (worst) to 1 + (best). An AMRI value of 0 is equivalent to random guessing or assigning the same score to all + edges in the candidate set. For more details on these metrics see :ref:`link-prediction-evaluation-metrics`. - Yaml: ``eval_metric:`` | ``- accuracy`` diff --git a/python/graphstorm/eval/evaluator.py b/python/graphstorm/eval/evaluator.py index 875cd1f46d..d32c29a9ec 100644 --- a/python/graphstorm/eval/evaluator.py +++ b/python/graphstorm/eval/evaluator.py @@ -1018,6 +1018,8 @@ def compute_score( # compute ranking value for each metric metrics: Dict[str, th.Tensor] = {} for metric in self.metric_list: + # NOTE: If other metrics needs candidate list sizes, add them here. + # Avoid adding the size twice to avoid possible errors. if metric == "amri": assert candidate_sizes, \ f"candidate_sizes needs to have a value for AMRI, got {candidate_sizes=}." diff --git a/python/graphstorm/run/gsgnn_lp/lp_infer_gnn.py b/python/graphstorm/run/gsgnn_lp/lp_infer_gnn.py index 36e10b0aa6..29b1907555 100644 --- a/python/graphstorm/run/gsgnn_lp/lp_infer_gnn.py +++ b/python/graphstorm/run/gsgnn_lp/lp_infer_gnn.py @@ -33,7 +33,7 @@ get_lm_ntypes, use_wholegraph, ) -from graphstorm.eval.eval_func import SUPPORTED_HIT_AT_METRICS +from graphstorm.eval.eval_func import SUPPORTED_HIT_AT_METRICS, SUPPORTED_LINK_PREDICTION_METRICS def main(config_args): """ main function @@ -56,10 +56,12 @@ def main(config_args): model_layer_to_load=config.restore_model_layers) infer = GSgnnLinkPredictionInferrer(model) infer.setup_device(device=get_device()) - assert all((x.startswith(SUPPORTED_HIT_AT_METRICS) or x == 'mrr') for x in - config.eval_metric), ( + assert all((x.startswith(SUPPORTED_HIT_AT_METRICS) + or x in SUPPORTED_LINK_PREDICTION_METRICS) + for x in config.eval_metric), ( "Invalid LP evaluation metrics. " - "GraphStorm only supports MRR and Hit@K metrics for link prediction.") + f"GraphStorm only supports {SUPPORTED_LINK_PREDICTION_METRICS} " + "and Hit@K metrics for link prediction.") if not config.no_validation: infer_idxs = infer_data.get_edge_test_set(config.eval_etype) infer.setup_evaluator(GSgnnLPEvaluator( diff --git a/python/graphstorm/run/gsgnn_lp/lp_infer_lm.py b/python/graphstorm/run/gsgnn_lp/lp_infer_lm.py index 4004556cd3..4640d98055 100644 --- a/python/graphstorm/run/gsgnn_lp/lp_infer_lm.py +++ b/python/graphstorm/run/gsgnn_lp/lp_infer_lm.py @@ -31,7 +31,7 @@ from graphstorm.dataloading import BUILTIN_LP_UNIFORM_NEG_SAMPLER from graphstorm.dataloading import BUILTIN_LP_JOINT_NEG_SAMPLER from graphstorm.utils import get_device -from graphstorm.eval.eval_func import SUPPORTED_HIT_AT_METRICS +from graphstorm.eval.eval_func import SUPPORTED_HIT_AT_METRICS, SUPPORTED_LINK_PREDICTION_METRICS def main(config_args): """ main function @@ -50,10 +50,12 @@ def main(config_args): model_layer_to_load=config.restore_model_layers) infer = GSgnnLinkPredictionInferrer(model) infer.setup_device(device=get_device()) - assert all((x.startswith(SUPPORTED_HIT_AT_METRICS) or x == 'mrr') for x in - config.eval_metric), ( + assert all((x.startswith(SUPPORTED_HIT_AT_METRICS) + or x in SUPPORTED_LINK_PREDICTION_METRICS) + for x in config.eval_metric), ( "Invalid LP evaluation metrics. " - "GraphStorm only supports MRR and Hit@K metrics for link prediction.") + f"GraphStorm only supports {SUPPORTED_LINK_PREDICTION_METRICS} " + "and Hit@K metrics for link prediction.") if not config.no_validation: infer_idxs = infer_data.get_edge_test_set(config.eval_etype) infer.setup_evaluator(GSgnnLPEvaluator( diff --git a/tests/end2end-tests/graphstorm-lp/mgpu_test.sh b/tests/end2end-tests/graphstorm-lp/mgpu_test.sh index 33338418e9..19c34f00e6 100644 --- a/tests/end2end-tests/graphstorm-lp/mgpu_test.sh +++ b/tests/end2end-tests/graphstorm-lp/mgpu_test.sh @@ -378,8 +378,8 @@ then exit -1 fi -echo "**************dataset: Movielens, do inference on saved model, decoder: dot" -python3 -m graphstorm.run.gs_link_prediction --inference --workspace $GS_HOME/inference_scripts/lp_infer --num-trainers $NUM_INFO_TRAINERS --num-servers 1 --num-samplers 0 --part-config /data/movielen_100k_lp_train_val_1p_4t/movie-lens-100k.json --ip-config ip_list.txt --ssh-port 2222 --cf ml_lp_infer.yaml --fanout '10,15' --num-layers 2 --use-mini-batch-infer false --use-node-embeddings true --eval-batch-size 1024 --save-embed-path /data/gsgnn_lp_ml_dot/infer-emb/ --restore-model-path /data/gsgnn_lp_ml_dot/epoch-$best_epoch_dot/ --logging-file /tmp/log.txt --preserve-input True +echo "**************dataset: Movielens, do inference on saved model, decoder: dot, metrics: mrr amri" +python3 -m graphstorm.run.gs_link_prediction --inference --workspace $GS_HOME/inference_scripts/lp_infer --num-trainers $NUM_INFO_TRAINERS --num-servers 1 --num-samplers 0 --part-config /data/movielen_100k_lp_train_val_1p_4t/movie-lens-100k.json --ip-config ip_list.txt --ssh-port 2222 --cf ml_lp_infer.yaml --fanout '10,15' --num-layers 2 --use-mini-batch-infer false --use-node-embeddings true --eval-batch-size 1024 --save-embed-path /data/gsgnn_lp_ml_dot/infer-emb/ --restore-model-path /data/gsgnn_lp_ml_dot/epoch-$best_epoch_dot/ --logging-file /tmp/log.txt --preserve-input True --eval-metric mrr amri error_and_exit $? @@ -390,6 +390,13 @@ then exit -1 fi +cnt=$(grep -c "| Test amri" /tmp/log.txt) +if test $cnt -ne 1 +then + echo "We do test, should have amri" + exit 1 +fi + bst_cnt=$(grep "Best Test mrr" /tmp/log.txt | wc -l) if test $bst_cnt -lt 1 then