From 407c17d3ecf766c8e37a2b7e4b9b1132136c4a33 Mon Sep 17 00:00:00 2001 From: Jannis Pohlmann Date: Wed, 25 Oct 2023 14:37:06 +0200 Subject: [PATCH] test: add tests for subgraph client fallback logic --- common/src/subgraph_client/client.rs | 218 +++++++++++++++++++++++++++ 1 file changed, 218 insertions(+) diff --git a/common/src/subgraph_client/client.rs b/common/src/subgraph_client/client.rs index 5d820d91..d893a6c2 100644 --- a/common/src/subgraph_client/client.rs +++ b/common/src/subgraph_client/client.rs @@ -134,6 +134,8 @@ impl SubgraphClient { #[cfg(test)] mod test { + use std::str::FromStr; + use serde_json::json; use wiremock::matchers::{method, path}; use wiremock::{Mock, MockServer, ResponseTemplate}; @@ -197,4 +199,220 @@ mod test { assert!(result.data.is_some()); } + + #[tokio::test] + async fn test_uses_local_deployment_if_healthy_and_synced() { + let deployment = + DeploymentId::from_str("QmAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA").unwrap(); + + let mock_server_local = MockServer::start().await; + mock_server_local + .register( + Mock::given(method("POST")) + .and(path("/status")) + .respond_with(ResponseTemplate::new(200).set_body_json(json!({ + "data": { + "indexingStatuses": [ + { + "synced": true, + "health": "healthy" + } + ] + } + }))), + ) + .await; + mock_server_local + .register( + Mock::given(method("POST")) + .and(path(&format!("/subgraphs/id/{}", deployment))) + .respond_with(ResponseTemplate::new(200).set_body_json(json!({ + "data": { + "user": { + "name": "local" + } + } + }))), + ) + .await; + + let mock_server_remote = MockServer::start().await; + mock_server_remote + .register( + Mock::given(method("POST")) + .and(path(&format!("/subgraphs/id/{}", deployment))) + .respond_with(ResponseTemplate::new(200).set_body_json(json!({ + "data": { + "user": { + "name": "remote" + } + } + }))), + ) + .await; + + // Create the subgraph client + let client = SubgraphClient::new( + Some(DeploymentDetails::for_graph_node(&mock_server_local.uri(), deployment).unwrap()), + DeploymentDetails::for_query_url(&format!( + "{}/subgraphs/id/{}", + mock_server_remote.uri(), + deployment + )) + .unwrap(), + ) + .unwrap(); + + // Query the subgraph + let response: Response = client + .query(&json!({ "query": "{ user(id: 1} { name } }"})) + .await + .unwrap(); + + assert_eq!(response.data, Some(json!({ "user": { "name": "local" } }))); + } + + #[tokio::test] + async fn test_uses_query_url_if_local_deployment_is_unhealthy() { + let deployment = + DeploymentId::from_str("QmAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA").unwrap(); + + let mock_server_local = MockServer::start().await; + mock_server_local + .register( + Mock::given(method("POST")) + .and(path("/status")) + .respond_with(ResponseTemplate::new(200).set_body_json(json!({ + "data": { + "indexingStatuses": [ + { + "synced": true, + "health": "unhealthy" + } + ] + } + }))), + ) + .await; + mock_server_local + .register( + Mock::given(method("POST")) + .and(path(&format!("/subgraphs/id/{}", deployment))) + .respond_with(ResponseTemplate::new(200).set_body_json(json!({ + "data": { + "user": { + "name": "local" + } + } + }))), + ) + .await; + + let mock_server_remote = MockServer::start().await; + mock_server_remote + .register( + Mock::given(method("POST")) + .and(path(&format!("/subgraphs/id/{}", deployment))) + .respond_with(ResponseTemplate::new(200).set_body_json(json!({ + "data": { + "user": { + "name": "remote" + } + } + }))), + ) + .await; + + // Create the subgraph client + let client = SubgraphClient::new( + Some(DeploymentDetails::for_graph_node(&mock_server_local.uri(), deployment).unwrap()), + DeploymentDetails::for_query_url(&format!( + "{}/subgraphs/id/{}", + mock_server_remote.uri(), + deployment + )) + .unwrap(), + ) + .unwrap(); + + // Query the subgraph + let response: Response = client + .query(&json!({ "query": "{ user(id: 1} { name } }"})) + .await + .unwrap(); + + assert_eq!(response.data, Some(json!({ "user": { "name": "remote" } }))); + } + + #[tokio::test] + async fn test_uses_query_url_if_local_deployment_is_not_synced() { + let deployment = + DeploymentId::from_str("QmAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA").unwrap(); + + let mock_server_local = MockServer::start().await; + mock_server_local + .register( + Mock::given(method("POST")) + .and(path("/status")) + .respond_with(ResponseTemplate::new(200).set_body_json(json!({ + "data": { + "indexingStatuses": [ + { + "synced": false, + "health": "healthy" + } + ] + } + }))), + ) + .await; + mock_server_local + .register( + Mock::given(method("POST")) + .and(path(&format!("/subgraphs/id/{}", deployment))) + .respond_with(ResponseTemplate::new(200).set_body_json(json!({ + "data": { + "user": { + "name": "local" + } + } + }))), + ) + .await; + + let mock_server_remote = MockServer::start().await; + mock_server_remote + .register( + Mock::given(method("POST")) + .and(path(&format!("/subgraphs/id/{}", deployment))) + .respond_with(ResponseTemplate::new(200).set_body_json(json!({ + "data": { + "user": { + "name": "remote" + } + } + }))), + ) + .await; + + // Create the subgraph client + let client = SubgraphClient::new( + Some(DeploymentDetails::for_graph_node(&mock_server_local.uri(), deployment).unwrap()), + DeploymentDetails::for_query_url(&format!( + "{}/subgraphs/id/{}", + mock_server_remote.uri(), + deployment + )) + .unwrap(), + ) + .unwrap(); + + // Query the subgraph + let response: Response = client + .query(&json!({ "query": "{ user(id: 1} { name } }"})) + .await + .unwrap(); + + assert_eq!(response.data, Some(json!({ "user": { "name": "remote" } }))); + } }