Skip to content

Commit

Permalink
Revert "Revert "Supergraph coprocessor implementation (#3408)" (#3604)"
Browse files Browse the repository at this point in the history
This reverts commit b1a2310.
  • Loading branch information
Geal committed Aug 22, 2023
1 parent 41aab27 commit 9e58a63
Show file tree
Hide file tree
Showing 8 changed files with 1,105 additions and 21 deletions.
33 changes: 27 additions & 6 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,27 @@ Note that the name of this metric may change in the future.

By [@Geal](https://github.com/Geal) in https://github.com/apollographql/router/pull/3513

### Supergraph coprocessor implementation ([PR #3408](https://github.com/apollographql/router/pull/3408))

Coprocessors now support supergraph service interception.

Supergraph request contains:
* query
* operation name
* variables
* extensions

Supergraph reqponse contains:
* label
* data
* path
* errors
* extensions

When using `@defer` or subscriptions a supergraph response may contain multiple GraphQL responses, and the coprocessor will be called for each.

By [@Geal](https://github.com/Geal) in https://github.com/apollographql/router/pull/3408

### Configure AWS SigV4 authentication for subgraph requests ([PR #3365](https://github.com/apollographql/router/pull/3365))

Secure your router to subgraph communication on AWS using [Signature Version 4](https://docs.aws.amazon.com/AmazonS3/latest/API/sig-v4-authenticating-requests.html) (Sigv4)!
Expand Down Expand Up @@ -87,7 +108,7 @@ By [@garypen](https://github.com/garypen) in https://github.com/apollographql/ro

### Spelling of `content_negociation` corrected to `content_negotiation` ([Issue #3204](https://github.com/apollographql/router/issues/3204))

We had a bit of a French twist on one of our internal module names. We won't promise it won't happen again, but `content_negociation` is spelled as `content_negotiation` now. 😄
We had a bit of a French twist on one of our internal module names. We won't promise it won't happen again, but `content_negociation` is spelled as `content_negotiation` now. 😄

Thank you for this contribution!

Expand Down Expand Up @@ -1443,7 +1464,7 @@ By [@Geal](https://github.com/Geal) in https://github.com/apollographql/router/p

### Small gzip'd responses no longer cause a panic

A regression introduced in v1.17.0 — again related to compression — has been resolved. This occurred when small responses used invalid buffer management, causing a panic.
A regression introduced in v1.17.0 — again related to compression — has been resolved. This occurred when small responses used invalid buffer management, causing a panic.

By [@dbanty](https://github.com/dbanty) in https://github.com/apollographql/router/pull/3047

Expand Down Expand Up @@ -1977,7 +1998,7 @@ By [@BrynCooke](https://github.com/BrynCooke) in https://github.com/apollographq

### Distributed caching: Don't send Redis' `CLIENT SETNAME` ([PR #2825](https://github.com/apollographql/router/pull/2825))

We won't send [the `CLIENT SETNAME` command](https://redis.io/commands/client-setname/) to connected Redis servers. This resolves an incompatibility with some Redis-compatible servers since not all "Redis-compatible" offerings (like Google Memorystore) actually support _every_ Redis command. We weren't actually necessitating this feature, it was just a feature that could be enabled optionally on our Redis client. No Router functionality is impacted.
We won't send [the `CLIENT SETNAME` command](https://redis.io/commands/client-setname/) to connected Redis servers. This resolves an incompatibility with some Redis-compatible servers since not all "Redis-compatible" offerings (like Google Memorystore) actually support _every_ Redis command. We weren't actually necessitating this feature, it was just a feature that could be enabled optionally on our Redis client. No Router functionality is impacted.

By [@Geal](https://github.com/Geal) in https://github.com/apollographql/router/pull/2825

Expand Down Expand Up @@ -3147,7 +3168,7 @@ By [@Geal](https://github.com/geal) in https://github.com/apollographql/router/p

### Optimize header propagation plugin's regular expression matching ([PR #2392](https://github.com/apollographql/router/pull/2392))

We've changed the header propagation plugins' behavior to reduce the chance of memory allocations occurring when applying regex-based header propagation rules.
We've changed the header propagation plugins' behavior to reduce the chance of memory allocations occurring when applying regex-based header propagation rules.

By [@o0Ignition0o](https://github.com/o0Ignition0o) in https://github.com/apollographql/router/pull/2392

Expand Down Expand Up @@ -3197,7 +3218,7 @@ By [@Geal](https://github.com/geal) in https://github.com/apollographql/router/p

Configuration changes will be [automatically migrated on load](https://www.apollographql.com/docs/router/configuration/overview#upgrading-your-router-configuration). However, you should update your source configuration files as these will become breaking changes in a future major release.

### Defer support graduates from preview ([Issue #2368](https://github.com/apollographql/router/issues/2368))
### Defer support graduates from preview ([Issue #2368](https://github.com/apollographql/router/issues/2368))

We're pleased to announce that [`@defer` support](https://www.apollographql.com/docs/router/executing-operations/defer-support/) has been promoted to general availability in accordance with our [product launch stages](https://www.apollographql.com/docs/resources/product-launch-stages/).

Expand Down Expand Up @@ -9136,4 +9157,4 @@ See our [release stages] for more information.

But the lack of clarity goes back to not having kept track of everything thus far! We can _fix_ our processes to keep track of these things! :smile_cat:

# [0.1.0] - TBA
# [0.1.0] - TBA
Original file line number Diff line number Diff line change
Expand Up @@ -770,6 +770,113 @@ expression: "&schema"
},
"additionalProperties": false
},
"supergraph": {
"description": "The supergraph stage request/response configuration",
"default": {
"request": {
"headers": false,
"context": false,
"body": false,
"sdl": false,
"path": false,
"method": false
},
"response": {
"headers": false,
"context": false,
"body": false,
"sdl": false,
"status_code": false
}
},
"type": "object",
"properties": {
"request": {
"description": "The request configuration",
"default": {
"headers": false,
"context": false,
"body": false,
"sdl": false,
"path": false,
"method": false
},
"type": "object",
"properties": {
"body": {
"description": "Send the body",
"default": false,
"type": "boolean"
},
"context": {
"description": "Send the context",
"default": false,
"type": "boolean"
},
"headers": {
"description": "Send the headers",
"default": false,
"type": "boolean"
},
"method": {
"description": "Send the method",
"default": false,
"type": "boolean"
},
"path": {
"description": "Send the path",
"default": false,
"type": "boolean"
},
"sdl": {
"description": "Send the SDL",
"default": false,
"type": "boolean"
}
},
"additionalProperties": false
},
"response": {
"description": "What information is passed to a router request/response stage",
"default": {
"headers": false,
"context": false,
"body": false,
"sdl": false,
"status_code": false
},
"type": "object",
"properties": {
"body": {
"description": "Send the body",
"default": false,
"type": "boolean"
},
"context": {
"description": "Send the context",
"default": false,
"type": "boolean"
},
"headers": {
"description": "Send the headers",
"default": false,
"type": "boolean"
},
"sdl": {
"description": "Send the SDL",
"default": false,
"type": "boolean"
},
"status_code": {
"description": "Send the HTTP status",
"default": false,
"type": "boolean"
}
},
"additionalProperties": false
}
}
},
"timeout": {
"description": "The timeout for external requests",
"default": {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ use crate::layers::ServiceBuilderExt;
use crate::plugin::Plugin;
use crate::plugin::PluginInit;
use crate::register_plugin;
use crate::services;
use crate::services::external::Control;
use crate::services::external::Externalizable;
use crate::services::external::PipelineStep;
Expand All @@ -44,6 +45,11 @@ use crate::services::router;
use crate::services::subgraph;
use crate::tracer::TraceId;

#[cfg(test)]
mod test;

mod supergraph;

pub(crate) const EXTERNAL_SPAN_NAME: &str = "external_plugin";
const POOL_IDLE_TIMEOUT_DURATION: Option<Duration> = Some(Duration::from_secs(5));

Expand Down Expand Up @@ -86,6 +92,13 @@ impl Plugin for CoprocessorPlugin<HTTPClientService> {
self.router_service(service)
}

fn supergraph_service(
&self,
service: services::supergraph::BoxService,
) -> services::supergraph::BoxService {
self.supergraph_service(service)
}

fn subgraph_service(&self, name: &str, service: subgraph::BoxService) -> subgraph::BoxService {
self.subgraph_service(name, service)
}
Expand Down Expand Up @@ -149,6 +162,18 @@ where
)
}

fn supergraph_service(
&self,
service: services::supergraph::BoxService,
) -> services::supergraph::BoxService {
self.configuration.supergraph.as_service(
self.http_client.clone(),
service,
self.configuration.url.clone(),
self.sdl.clone(),
)
}

fn subgraph_service(&self, name: &str, service: subgraph::BoxService) -> subgraph::BoxService {
self.configuration.subgraph.all.as_service(
self.http_client.clone(),
Expand Down Expand Up @@ -239,6 +264,9 @@ struct Conf {
/// The router stage request/response configuration
#[serde(default)]
router: RouterStage,
/// The supergraph stage request/response configuration
#[serde(default)]
supergraph: supergraph::SupergraphStage,
/// The subgraph stage request/response configuration
#[serde(default)]
subgraph: SubgraphStages,
Expand Down Expand Up @@ -676,7 +704,7 @@ where
// If first is None, or contains an error we return an error
let opt_first: Option<Bytes> = first.and_then(|f| f.ok());
let bytes = match opt_first {
Some(b) => b.to_vec(),
Some(b) => b,
None => {
tracing::error!(
"Coprocessor cannot convert body into future due to problem with first part"
Expand All @@ -695,7 +723,7 @@ where
.transpose()?;
let body_to_send = response_config
.body
.then(|| String::from_utf8(bytes.clone()))
.then(|| std::str::from_utf8(&bytes).map(|s| s.to_string()))
.transpose()?;
let status_to_send = response_config.status_code.then(|| parts.status.as_u16());
let context_to_send = response_config.context.then(|| response.context.clone());
Expand Down Expand Up @@ -857,7 +885,6 @@ where
// First, extract the data we need from our request and prepare our
// external call. Use our configuration to figure out which data to send.
let (parts, body) = request.subgraph_request.into_parts();
let bytes = Bytes::from(serde_json::to_vec(&body)?);

let headers_to_send = request_config
.headers
Expand All @@ -866,7 +893,7 @@ where

let body_to_send = request_config
.body
.then(|| serde_json::from_slice::<serde_json::Value>(&bytes))
.then(|| serde_json::to_value(&body))
.transpose()?;
let context_to_send = request_config.context.then(|| request.context.clone());
let uri = request_config.uri.then(|| parts.uri.to_string());
Expand Down Expand Up @@ -997,7 +1024,6 @@ where
// external call. Use our configuration to figure out which data to send.

let (parts, body) = response.response.into_parts();
let bytes = Bytes::from(serde_json::to_vec(&body)?);

let headers_to_send = response_config
.headers
Expand All @@ -1008,7 +1034,7 @@ where

let body_to_send = response_config
.body
.then(|| serde_json::from_slice::<serde_json::Value>(&bytes))
.then(|| serde_json::to_value(&body))
.transpose()?;
let context_to_send = response_config.context.then(|| response.context.clone());
let service_name = response_config.service_name.then_some(service_name);
Expand Down Expand Up @@ -1098,7 +1124,7 @@ fn validate_coprocessor_output<T>(
}

/// Convert a HeaderMap into a HashMap
pub(super) fn externalize_header_map(
pub(crate) fn externalize_header_map(
input: &HeaderMap<HeaderValue>,
) -> Result<HashMap<String, Vec<String>>, BoxError> {
let mut output = HashMap::new();
Expand Down
Loading

0 comments on commit 9e58a63

Please sign in to comment.