Skip to content

Commit

Permalink
feat: Re-genericize tokens (#318)
Browse files Browse the repository at this point in the history
* feat: Remake tokens generic
re: #221 #203

Turns out you can actually get around candid's limitation on `IDLValue`. It just takes some careful rejiggering.

First, use `Reserved` to match anything. This seems like a valid use of `Reserved`, and should basically always be supported.

Second, use `deserialize_ignored_any` to actually get all the content with `IDLValueVisitor`. This is a little bit of a hack, but it works.
  • Loading branch information
Daniel-Bloom-dfinity authored Mar 7, 2022
1 parent 784c460 commit db7886d
Show file tree
Hide file tree
Showing 3 changed files with 200 additions and 46 deletions.
9 changes: 6 additions & 3 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

7 changes: 5 additions & 2 deletions ic-utils/Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[package]
name = "ic-utils"
version = "0.12.1"
version = "0.12.2"
authors = ["DFINITY Stiftung <[email protected]>"]
edition = "2018"
description = "Collection of utilities for Rust, on top of ic-agent, to communicate with the Internet Computer, following the Public Specification."
Expand All @@ -16,14 +16,17 @@ include = ["src", "Cargo.toml", "../LICENSE", "README.md"]

[dependencies]
async-trait = "0.1.40"
candid = "0.7.10"
candid = "0.7.12"
garcon = { version = "0.2", features = ["async"] }
ic-agent = { path = "../ic-agent", version = "0.12" }
serde = "1.0.115"
serde_bytes = "0.11"
strum = "0.23"
strum_macros = "0.23"
thiserror = "1.0.29"
paste = "1"
num-bigint = "0.4"
leb128 = "0.2"

[dev-dependencies]
ring = "0.16.11"
Expand Down
230 changes: 189 additions & 41 deletions ic-utils/src/interfaces/http_request.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,16 @@
//! The canister interface for canisters that implement HTTP requests.

use crate::{call::AsyncCall, call::SyncCall, canister::CanisterBuilder, Canister};
use candid::{CandidType, Deserialize, Func, Nat};
use crate::{
call::{AsyncCall, SyncCall},
canister::CanisterBuilder,
Canister,
};
use candid::{
parser::value::{IDLValue, IDLValueVisitor},
types::{Serializer, Type},
CandidType, Deserialize, Func,
};
use ic_agent::{export::Principal, Agent};
use serde_bytes::ByteBuf;
use std::fmt::Debug;

/// A canister that can serve a HTTP request.
Expand All @@ -28,59 +35,72 @@ pub struct HttpRequest<'body> {
pub body: &'body [u8],
}

/// A token for continuing a callback streaming strategy.
#[derive(Debug, Clone, CandidType, Deserialize)]
pub struct Token {
key: String,
content_encoding: String,
index: Nat,
// The sha ensures that a client doesn't stream part of one version of an asset
// followed by part of a different asset, even if not checking the certificate.
sha256: Option<ByteBuf>,
}

/// A callback-token pair for a callback streaming strategy.
#[derive(Debug, Clone, CandidType, Deserialize)]
pub struct CallbackStrategy {
/// The callback function to be called to continue the stream.
pub callback: Func,
/// The token to pass to the function.
pub token: Token,
}

/// Possible strategies for a streaming response.
#[derive(Debug, Clone, CandidType, Deserialize)]
pub enum StreamingStrategy {
/// A callback-based streaming strategy, where a callback function is provided for continuing the stream.
Callback(CallbackStrategy),
}

/// A HTTP response.
#[derive(Debug, Clone, CandidType, Deserialize)]
pub struct HttpResponse {
pub struct HttpResponse<Token = self::Token> {
/// The HTTP status code.
pub status_code: u16,
/// The response header map.
pub headers: Vec<HeaderField>,
#[serde(with = "serde_bytes")]
/// The response body.
#[serde(with = "serde_bytes")]
pub body: Vec<u8>,
/// The strategy for streaming the rest of the data, if the full response is to be streamed.
pub streaming_strategy: Option<StreamingStrategy>,
pub streaming_strategy: Option<StreamingStrategy<Token>>,
/// Whether the query call should be upgraded to an update call.
pub upgrade: Option<bool>,
}

/// Possible strategies for a streaming response.
#[derive(Debug, Clone, CandidType, Deserialize)]
pub enum StreamingStrategy<Token = self::Token> {
/// A callback-based streaming strategy, where a callback function is provided for continuing the stream.
Callback(CallbackStrategy<Token>),
}

/// A callback-token pair for a callback streaming strategy.
#[derive(Debug, Clone, CandidType, Deserialize)]
pub struct CallbackStrategy<Token = self::Token> {
/// The callback function to be called to continue the stream.
pub callback: Func,
/// The token to pass to the function.
pub token: Token,
}

/// The next chunk of a streaming HTTP response.
#[derive(Debug, Clone, CandidType, Deserialize)]
pub struct StreamingCallbackHttpResponse {
pub struct StreamingCallbackHttpResponse<Token = self::Token> {
/// The body of the stream chunk.
#[serde(with = "serde_bytes")]
pub body: Vec<u8>,
/// The new stream continuation token.
pub token: Option<Token>,
}

/// A token for continuing a callback streaming strategy.
#[derive(Debug, Clone, PartialEq)]
pub struct Token(pub IDLValue);

impl CandidType for Token {
fn _ty() -> Type {
Type::Reserved
}
fn idl_serialize<S: Serializer>(&self, _serializer: S) -> Result<(), S::Error> {
// We cannot implement serialize, since our type must be `Reserved` in order to accept anything.
// Attempting to serialize this type is always an error and should be regarded as a compile time error.
unimplemented!("Token is not serializable")
}
}

impl<'de> Deserialize<'de> for Token {
fn deserialize<D: serde::de::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
// Ya know it says `ignored`, but what if we just didn't ignore it.
deserializer
.deserialize_ignored_any(IDLValueVisitor)
.map(Token)
}
}

impl HttpRequestCanister {
/// Create an instance of a [Canister] implementing the [HttpRequestCanister] interface
/// and pointing to the right Canister ID.
Expand Down Expand Up @@ -151,25 +171,30 @@ impl<'agent> Canister<'agent, HttpRequestCanister> {
method: M,
token: Token,
) -> impl 'agent + SyncCall<(StreamingCallbackHttpResponse,)> {
self.query_(&method.into()).with_arg(token).build()
self.query_(&method.into()).with_value_arg(token.0).build()
}
}

#[cfg(test)]
mod test {
use super::HttpResponse;
use candid::{Decode, Encode};
use super::{
CallbackStrategy, HttpResponse, StreamingCallbackHttpResponse, StreamingStrategy, Token,
};
use candid::{
parser::value::{IDLField, IDLValue},
Decode, Encode,
};

mod pre_update_legacy {
use candid::{CandidType, Deserialize, Func, Nat};
use serde_bytes::ByteBuf;

#[derive(CandidType, Deserialize)]
pub struct Token {
key: String,
content_encoding: String,
index: Nat,
sha256: Option<ByteBuf>,
pub key: String,
pub content_encoding: String,
pub index: Nat,
pub sha256: Option<ByteBuf>,
}

#[derive(CandidType, Deserialize)]
Expand Down Expand Up @@ -208,4 +233,127 @@ mod test {

let _response = Decode!(&bytes, HttpResponse).unwrap();
}

#[test]
fn deserialize_response_with_token() {
use candid::{types::Label, Func, Principal};

let bytes: Vec<u8> = Encode!(&HttpResponse {
status_code: 100,
headers: Vec::new(),
body: Vec::new(),
streaming_strategy: Some(StreamingStrategy::Callback(CallbackStrategy {
callback: Func {
principal: Principal::from_text("2chl6-4hpzw-vqaaa-aaaaa-c").unwrap(),
method: "callback".into()
},
token: pre_update_legacy::Token {
key: "foo".into(),
content_encoding: "bar".into(),
index: 42.into(),
sha256: None,
},
})),
upgrade: None,
})
.unwrap();

let response = Decode!(&bytes, HttpResponse).unwrap();
assert_eq!(response.status_code, 100);
let token = match response.streaming_strategy {
Some(StreamingStrategy::Callback(CallbackStrategy { token, .. })) => token,
_ => panic!("streaming_strategy was missing"),
};
let fields = match token {
Token(IDLValue::Record(fields)) => fields,
_ => panic!("token type mismatched {:?}", token),
};
assert!(fields.contains(&IDLField {
id: Label::Named("key".into()),
val: IDLValue::Text("foo".into())
}));
assert!(fields.contains(&IDLField {
id: Label::Named("content_encoding".into()),
val: IDLValue::Text("bar".into())
}));
assert!(fields.contains(&IDLField {
id: Label::Named("index".into()),
val: IDLValue::Nat(42.into())
}));
assert!(fields.contains(&IDLField {
id: Label::Named("sha256".into()),
val: IDLValue::None
}));
}

#[test]
fn deserialize_streaming_response_with_token() {
use candid::types::Label;

let bytes: Vec<u8> = Encode!(&StreamingCallbackHttpResponse {
body: b"this is a body".as_ref().into(),
token: Some(pre_update_legacy::Token {
key: "foo".into(),
content_encoding: "bar".into(),
index: 42.into(),
sha256: None,
}),
})
.unwrap();

let response = Decode!(&bytes, StreamingCallbackHttpResponse).unwrap();
assert_eq!(response.body, b"this is a body");
let fields = match response.token {
Some(Token(IDLValue::Record(fields))) => fields,
_ => panic!("token type mismatched {:?}", response.token),
};
assert!(fields.contains(&IDLField {
id: Label::Named("key".into()),
val: IDLValue::Text("foo".into())
}));
assert!(fields.contains(&IDLField {
id: Label::Named("content_encoding".into()),
val: IDLValue::Text("bar".into())
}));
assert!(fields.contains(&IDLField {
id: Label::Named("index".into()),
val: IDLValue::Nat(42.into())
}));
assert!(fields.contains(&IDLField {
id: Label::Named("sha256".into()),
val: IDLValue::None
}));
}

#[test]
fn deserialize_streaming_response_without_token() {
mod missing_token {
use candid::{CandidType, Deserialize};
/// The next chunk of a streaming HTTP response.
#[derive(Debug, Clone, CandidType, Deserialize)]
pub struct StreamingCallbackHttpResponse {
/// The body of the stream chunk.
#[serde(with = "serde_bytes")]
pub body: Vec<u8>,
}
}
let bytes: Vec<u8> = Encode!(&missing_token::StreamingCallbackHttpResponse {
body: b"this is a body".as_ref().into(),
})
.unwrap();

let response = Decode!(&bytes, StreamingCallbackHttpResponse).unwrap();
assert_eq!(response.body, b"this is a body");
assert_eq!(response.token, None);

let bytes: Vec<u8> = Encode!(&StreamingCallbackHttpResponse {
body: b"this is a body".as_ref().into(),
token: Option::<pre_update_legacy::Token>::None,
})
.unwrap();

let response = Decode!(&bytes, StreamingCallbackHttpResponse).unwrap();
assert_eq!(response.body, b"this is a body");
assert_eq!(response.token, None);
}
}

0 comments on commit db7886d

Please sign in to comment.