//! Generated by `trust-tasks-codegen` — do not edit by hand.
//!
//! Spec slug: `consent/request`. Version: `1.0`.
#[allow(unused_imports)]
use serde::{Deserialize, Serialize};
/// Error types.
pub mod error {
/// Error from a `TryFrom` or `FromStr` implementation.
pub struct ConversionError(::std::borrow::Cow<'static, str>);
impl ::std::error::Error for ConversionError {}
impl ::std::fmt::Display for ConversionError {
fn fmt(&self, f: &mut ::std::fmt::Formatter<'_>) -> Result<(), ::std::fmt::Error> {
::std::fmt::Display::fmt(&self.0, f)
}
}
impl ::std::fmt::Debug for ConversionError {
fn fmt(&self, f: &mut ::std::fmt::Formatter<'_>) -> Result<(), ::std::fmt::Error> {
::std::fmt::Debug::fmt(&self.0, f)
}
}
impl From<&'static str> for ConversionError {
fn from(value: &'static str) -> Self {
Self(value.into())
}
}
impl From<String> for ConversionError {
fn from(value: String) -> Self {
Self(value.into())
}
}
}
///The platform-agnostic identifier of WHAT is being consented to: one conversation, for one agent.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "The platform-agnostic identifier of WHAT is being consented to: one conversation, for one agent.",
/// "type": "object",
/// "required": [
/// "agent",
/// "conversationRef",
/// "kind",
/// "platform"
/// ],
/// "properties": {
/// "agent": {
/// "description": "VID (DID) of the AI agent the conversation would reach.",
/// "type": "string",
/// "minLength": 1
/// },
/// "conversationRef": {
/// "description": "The bridge's OPAQUE conversation handle (e.g. \"sig-1a2b3c4d\"). NEVER the raw platform address — the VTA never learns the phone number / chat id.",
/// "type": "string",
/// "minLength": 1
/// },
/// "kind": {
/// "$ref": "#/definitions/Kind"
/// },
/// "platform": {
/// "description": "Messaging-platform tag, e.g. \"signal\", \"whatsapp\", \"slack\".",
/// "type": "string",
/// "minLength": 1
/// }
/// },
/// "additionalProperties": false
///}
/// ```
/// </details>
#[derive(::serde::Deserialize, ::serde::Serialize, Clone, Debug)]
#[serde(deny_unknown_fields)]
pub struct ConsentSubject {
///VID (DID) of the AI agent the conversation would reach.
pub agent: ConsentSubjectAgent,
///The bridge's OPAQUE conversation handle (e.g. "sig-1a2b3c4d"). NEVER the raw platform address — the VTA never learns the phone number / chat id.
#[serde(rename = "conversationRef")]
pub conversation_ref: ConsentSubjectConversationRef,
pub kind: Kind,
///Messaging-platform tag, e.g. "signal", "whatsapp", "slack".
pub platform: ConsentSubjectPlatform,
}
///VID (DID) of the AI agent the conversation would reach.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "VID (DID) of the AI agent the conversation would reach.",
/// "type": "string",
/// "minLength": 1
///}
/// ```
/// </details>
#[derive(::serde::Serialize, Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[serde(transparent)]
pub struct ConsentSubjectAgent(::std::string::String);
impl ::std::ops::Deref for ConsentSubjectAgent {
type Target = ::std::string::String;
fn deref(&self) -> &::std::string::String {
&self.0
}
}
impl ::std::convert::From<ConsentSubjectAgent> for ::std::string::String {
fn from(value: ConsentSubjectAgent) -> Self {
value.0
}
}
impl ::std::str::FromStr for ConsentSubjectAgent {
type Err = self::error::ConversionError;
fn from_str(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
if value.chars().count() < 1usize {
return Err("shorter than 1 characters".into());
}
Ok(Self(value.to_string()))
}
}
impl ::std::convert::TryFrom<&str> for ConsentSubjectAgent {
type Error = self::error::ConversionError;
fn try_from(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<&::std::string::String> for ConsentSubjectAgent {
type Error = self::error::ConversionError;
fn try_from(
value: &::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<::std::string::String> for ConsentSubjectAgent {
type Error = self::error::ConversionError;
fn try_from(
value: ::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl<'de> ::serde::Deserialize<'de> for ConsentSubjectAgent {
fn deserialize<D>(deserializer: D) -> ::std::result::Result<Self, D::Error>
where
D: ::serde::Deserializer<'de>,
{
::std::string::String::deserialize(deserializer)?
.parse()
.map_err(|e: self::error::ConversionError| {
<D::Error as ::serde::de::Error>::custom(e.to_string())
})
}
}
///The bridge's OPAQUE conversation handle (e.g. "sig-1a2b3c4d"). NEVER the raw platform address — the VTA never learns the phone number / chat id.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "The bridge's OPAQUE conversation handle (e.g. \"sig-1a2b3c4d\"). NEVER the raw platform address — the VTA never learns the phone number / chat id.",
/// "type": "string",
/// "minLength": 1
///}
/// ```
/// </details>
#[derive(::serde::Serialize, Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[serde(transparent)]
pub struct ConsentSubjectConversationRef(::std::string::String);
impl ::std::ops::Deref for ConsentSubjectConversationRef {
type Target = ::std::string::String;
fn deref(&self) -> &::std::string::String {
&self.0
}
}
impl ::std::convert::From<ConsentSubjectConversationRef> for ::std::string::String {
fn from(value: ConsentSubjectConversationRef) -> Self {
value.0
}
}
impl ::std::str::FromStr for ConsentSubjectConversationRef {
type Err = self::error::ConversionError;
fn from_str(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
if value.chars().count() < 1usize {
return Err("shorter than 1 characters".into());
}
Ok(Self(value.to_string()))
}
}
impl ::std::convert::TryFrom<&str> for ConsentSubjectConversationRef {
type Error = self::error::ConversionError;
fn try_from(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<&::std::string::String> for ConsentSubjectConversationRef {
type Error = self::error::ConversionError;
fn try_from(
value: &::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<::std::string::String> for ConsentSubjectConversationRef {
type Error = self::error::ConversionError;
fn try_from(
value: ::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl<'de> ::serde::Deserialize<'de> for ConsentSubjectConversationRef {
fn deserialize<D>(deserializer: D) -> ::std::result::Result<Self, D::Error>
where
D: ::serde::Deserializer<'de>,
{
::std::string::String::deserialize(deserializer)?
.parse()
.map_err(|e: self::error::ConversionError| {
<D::Error as ::serde::de::Error>::custom(e.to_string())
})
}
}
///Messaging-platform tag, e.g. "signal", "whatsapp", "slack".
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "Messaging-platform tag, e.g. \"signal\", \"whatsapp\", \"slack\".",
/// "type": "string",
/// "minLength": 1
///}
/// ```
/// </details>
#[derive(::serde::Serialize, Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[serde(transparent)]
pub struct ConsentSubjectPlatform(::std::string::String);
impl ::std::ops::Deref for ConsentSubjectPlatform {
type Target = ::std::string::String;
fn deref(&self) -> &::std::string::String {
&self.0
}
}
impl ::std::convert::From<ConsentSubjectPlatform> for ::std::string::String {
fn from(value: ConsentSubjectPlatform) -> Self {
value.0
}
}
impl ::std::str::FromStr for ConsentSubjectPlatform {
type Err = self::error::ConversionError;
fn from_str(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
if value.chars().count() < 1usize {
return Err("shorter than 1 characters".into());
}
Ok(Self(value.to_string()))
}
}
impl ::std::convert::TryFrom<&str> for ConsentSubjectPlatform {
type Error = self::error::ConversionError;
fn try_from(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<&::std::string::String> for ConsentSubjectPlatform {
type Error = self::error::ConversionError;
fn try_from(
value: &::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<::std::string::String> for ConsentSubjectPlatform {
type Error = self::error::ConversionError;
fn try_from(
value: ::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl<'de> ::serde::Deserialize<'de> for ConsentSubjectPlatform {
fn deserialize<D>(deserializer: D) -> ::std::result::Result<Self, D::Error>
where
D: ::serde::Deserializer<'de>,
{
::std::string::String::deserialize(deserializer)?
.parse()
.map_err(|e: self::error::ConversionError| {
<D::Error as ::serde::de::Error>::custom(e.to_string())
})
}
}
///Vendor-namespaced extension object per SPEC.md §4.5.1. Each immediate key MUST be a reverse-DNS namespace; structure under each namespace is opaque to the framework.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "title": "Ext",
/// "description": "Vendor-namespaced extension object per SPEC.md §4.5.1. Each immediate key MUST be a reverse-DNS namespace; structure under each namespace is opaque to the framework.",
/// "type": "object",
/// "minProperties": 1,
/// "additionalProperties": true,
/// "propertyNames": {
/// "pattern": "^[a-z][a-z0-9-]*(\\.[a-z0-9-]+)+$"
/// }
///}
/// ```
/// </details>
#[derive(::serde::Deserialize, ::serde::Serialize, Clone, Debug)]
#[serde(transparent)]
pub struct Ext(pub ::std::collections::HashMap<ExtKey, ::serde_json::Value>);
impl ::std::ops::Deref for Ext {
type Target = ::std::collections::HashMap<ExtKey, ::serde_json::Value>;
fn deref(&self) -> &::std::collections::HashMap<ExtKey, ::serde_json::Value> {
&self.0
}
}
impl ::std::convert::From<Ext> for ::std::collections::HashMap<ExtKey, ::serde_json::Value> {
fn from(value: Ext) -> Self {
value.0
}
}
impl ::std::convert::From<::std::collections::HashMap<ExtKey, ::serde_json::Value>> for Ext {
fn from(value: ::std::collections::HashMap<ExtKey, ::serde_json::Value>) -> Self {
Self(value)
}
}
///`ExtKey`
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "type": "string",
/// "pattern": "^[a-z][a-z0-9-]*(\\.[a-z0-9-]+)+$"
///}
/// ```
/// </details>
#[derive(::serde::Serialize, Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[serde(transparent)]
pub struct ExtKey(::std::string::String);
impl ::std::ops::Deref for ExtKey {
type Target = ::std::string::String;
fn deref(&self) -> &::std::string::String {
&self.0
}
}
impl ::std::convert::From<ExtKey> for ::std::string::String {
fn from(value: ExtKey) -> Self {
value.0
}
}
impl ::std::str::FromStr for ExtKey {
type Err = self::error::ConversionError;
fn from_str(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
static PATTERN: ::std::sync::LazyLock<::regress::Regex> =
::std::sync::LazyLock::new(|| {
::regress::Regex::new("^[a-z][a-z0-9-]*(\\.[a-z0-9-]+)+$").unwrap()
});
if PATTERN.find(value).is_none() {
return Err("doesn't match pattern \"^[a-z][a-z0-9-]*(\\.[a-z0-9-]+)+$\"".into());
}
Ok(Self(value.to_string()))
}
}
impl ::std::convert::TryFrom<&str> for ExtKey {
type Error = self::error::ConversionError;
fn try_from(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<&::std::string::String> for ExtKey {
type Error = self::error::ConversionError;
fn try_from(
value: &::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<::std::string::String> for ExtKey {
type Error = self::error::ConversionError;
fn try_from(
value: ::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl<'de> ::serde::Deserialize<'de> for ExtKey {
fn deserialize<D>(deserializer: D) -> ::std::result::Result<Self, D::Error>
where
D: ::serde::Deserializer<'de>,
{
::std::string::String::deserialize(deserializer)?
.parse()
.map_err(|e: self::error::ConversionError| {
<D::Error as ::serde::de::Error>::custom(e.to_string())
})
}
}
///The interaction kind: a 1:1 direct message, a multi-party group, or a broadcast channel.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "The interaction kind: a 1:1 direct message, a multi-party group, or a broadcast channel.",
/// "type": "string",
/// "enum": [
/// "dm",
/// "group",
/// "channel"
/// ]
///}
/// ```
/// </details>
#[derive(
::serde::Deserialize,
::serde::Serialize,
Clone,
Copy,
Debug,
Eq,
Hash,
Ord,
PartialEq,
PartialOrd,
)]
pub enum Kind {
#[serde(rename = "dm")]
Dm,
#[serde(rename = "group")]
Group,
#[serde(rename = "channel")]
Channel,
}
impl ::std::fmt::Display for Kind {
fn fmt(&self, f: &mut ::std::fmt::Formatter<'_>) -> ::std::fmt::Result {
match *self {
Self::Dm => f.write_str("dm"),
Self::Group => f.write_str("group"),
Self::Channel => f.write_str("channel"),
}
}
}
impl ::std::str::FromStr for Kind {
type Err = self::error::ConversionError;
fn from_str(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
match value {
"dm" => Ok(Self::Dm),
"group" => Ok(Self::Group),
"channel" => Ok(Self::Channel),
_ => Err("invalid value".into()),
}
}
}
impl ::std::convert::TryFrom<&str> for Kind {
type Error = self::error::ConversionError;
fn try_from(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<&::std::string::String> for Kind {
type Error = self::error::ConversionError;
fn try_from(
value: &::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<::std::string::String> for Kind {
type Error = self::error::ConversionError;
fn try_from(
value: ::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
///A bridge asks the VTA whether an inbound conversation may reach an AI agent, prompting operator consent on first contact.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "$id": "https://trusttasks.org/spec/consent/request/1.0",
/// "title": "Payload",
/// "description": "A bridge asks the VTA whether an inbound conversation may reach an AI agent, prompting operator consent on first contact.",
/// "type": "object",
/// "required": [
/// "challenge",
/// "scope",
/// "subject"
/// ],
/// "properties": {
/// "challenge": {
/// "description": "base64url-encoded nonce (≥128 bits entropy) echoed by the matching consent/decision, so the bridge can correlate the decision to this request.",
/// "type": "string",
/// "minLength": 16
/// },
/// "contextHint": {
/// "description": "Optional VTA context path the bridge runs under, to scope approver resolution.",
/// "type": "string"
/// },
/// "displayHint": {
/// "description": "Optional operator-facing label (e.g. \"Signal group 'Family'\"). MUST NOT contain a raw platform address.",
/// "type": "string"
/// },
/// "ext": {
/// "$ref": "#/definitions/Ext"
/// },
/// "firstMessageDigest": {
/// "description": "Optional multihash digest of the held first message, binding the request to concrete content.",
/// "type": "string"
/// },
/// "scope": {
/// "description": "The access the bridge seeks for the agent on this conversation.",
/// "$ref": "#/definitions/Scope"
/// },
/// "subject": {
/// "$ref": "#/definitions/ConsentSubject"
/// }
/// },
/// "additionalProperties": false
///}
/// ```
/// </details>
#[derive(::serde::Deserialize, ::serde::Serialize, Clone, Debug)]
#[serde(deny_unknown_fields)]
pub struct Payload {
///base64url-encoded nonce (≥128 bits entropy) echoed by the matching consent/decision, so the bridge can correlate the decision to this request.
pub challenge: PayloadChallenge,
///Optional VTA context path the bridge runs under, to scope approver resolution.
#[serde(
rename = "contextHint",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub context_hint: ::std::option::Option<::std::string::String>,
///Optional operator-facing label (e.g. "Signal group 'Family'"). MUST NOT contain a raw platform address.
#[serde(
rename = "displayHint",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub display_hint: ::std::option::Option<::std::string::String>,
#[serde(default, skip_serializing_if = "::std::option::Option::is_none")]
pub ext: ::std::option::Option<Ext>,
///Optional multihash digest of the held first message, binding the request to concrete content.
#[serde(
rename = "firstMessageDigest",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub first_message_digest: ::std::option::Option<::std::string::String>,
///The access the bridge seeks for the agent on this conversation.
pub scope: Scope,
pub subject: ConsentSubject,
}
///base64url-encoded nonce (≥128 bits entropy) echoed by the matching consent/decision, so the bridge can correlate the decision to this request.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "base64url-encoded nonce (≥128 bits entropy) echoed by the matching consent/decision, so the bridge can correlate the decision to this request.",
/// "type": "string",
/// "minLength": 16
///}
/// ```
/// </details>
#[derive(::serde::Serialize, Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[serde(transparent)]
pub struct PayloadChallenge(::std::string::String);
impl ::std::ops::Deref for PayloadChallenge {
type Target = ::std::string::String;
fn deref(&self) -> &::std::string::String {
&self.0
}
}
impl ::std::convert::From<PayloadChallenge> for ::std::string::String {
fn from(value: PayloadChallenge) -> Self {
value.0
}
}
impl ::std::str::FromStr for PayloadChallenge {
type Err = self::error::ConversionError;
fn from_str(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
if value.chars().count() < 16usize {
return Err("shorter than 16 characters".into());
}
Ok(Self(value.to_string()))
}
}
impl ::std::convert::TryFrom<&str> for PayloadChallenge {
type Error = self::error::ConversionError;
fn try_from(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<&::std::string::String> for PayloadChallenge {
type Error = self::error::ConversionError;
fn try_from(
value: &::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<::std::string::String> for PayloadChallenge {
type Error = self::error::ConversionError;
fn try_from(
value: ::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl<'de> ::serde::Deserialize<'de> for PayloadChallenge {
fn deserialize<D>(deserializer: D) -> ::std::result::Result<Self, D::Error>
where
D: ::serde::Deserializer<'de>,
{
::std::string::String::deserialize(deserializer)?
.parse()
.map_err(|e: self::error::ConversionError| {
<D::Error as ::serde::de::Error>::custom(e.to_string())
})
}
}
///Synchronous ack from the VTA: a prompt was routed (accepted) or the request was refused. The actual decision arrives out-of-band as a consent/decision.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "title": "Response",
/// "description": "Synchronous ack from the VTA: a prompt was routed (accepted) or the request was refused. The actual decision arrives out-of-band as a consent/decision.",
/// "type": "object",
/// "required": [
/// "status"
/// ],
/// "properties": {
/// "ext": {
/// "$ref": "#/definitions/Ext"
/// },
/// "reason": {
/// "description": "Required when status is `refused`.",
/// "type": "string"
/// },
/// "requestId": {
/// "description": "The VTA's id for the pending consent (set when accepted), for correlation and polling.",
/// "type": "string"
/// },
/// "status": {
/// "description": "`accepted` = a pending consent was minted and an approval prompt routed. `refused` = not actionable; `reason` MUST be set.",
/// "type": "string",
/// "enum": [
/// "accepted",
/// "refused"
/// ]
/// }
/// },
/// "additionalProperties": false,
/// "$anchor": "response"
///}
/// ```
/// </details>
#[derive(::serde::Deserialize, ::serde::Serialize, Clone, Debug)]
#[serde(deny_unknown_fields)]
pub struct Response {
#[serde(default, skip_serializing_if = "::std::option::Option::is_none")]
pub ext: ::std::option::Option<Ext>,
///Required when status is `refused`.
#[serde(default, skip_serializing_if = "::std::option::Option::is_none")]
pub reason: ::std::option::Option<::std::string::String>,
///The VTA's id for the pending consent (set when accepted), for correlation and polling.
#[serde(
rename = "requestId",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub request_id: ::std::option::Option<::std::string::String>,
///`accepted` = a pending consent was minted and an approval prompt routed. `refused` = not actionable; `reason` MUST be set.
pub status: ResponseStatus,
}
///`accepted` = a pending consent was minted and an approval prompt routed. `refused` = not actionable; `reason` MUST be set.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "`accepted` = a pending consent was minted and an approval prompt routed. `refused` = not actionable; `reason` MUST be set.",
/// "type": "string",
/// "enum": [
/// "accepted",
/// "refused"
/// ]
///}
/// ```
/// </details>
#[derive(
::serde::Deserialize,
::serde::Serialize,
Clone,
Copy,
Debug,
Eq,
Hash,
Ord,
PartialEq,
PartialOrd,
)]
pub enum ResponseStatus {
#[serde(rename = "accepted")]
Accepted,
#[serde(rename = "refused")]
Refused,
}
impl ::std::fmt::Display for ResponseStatus {
fn fmt(&self, f: &mut ::std::fmt::Formatter<'_>) -> ::std::fmt::Result {
match *self {
Self::Accepted => f.write_str("accepted"),
Self::Refused => f.write_str("refused"),
}
}
}
impl ::std::str::FromStr for ResponseStatus {
type Err = self::error::ConversionError;
fn from_str(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
match value {
"accepted" => Ok(Self::Accepted),
"refused" => Ok(Self::Refused),
_ => Err("invalid value".into()),
}
}
}
impl ::std::convert::TryFrom<&str> for ResponseStatus {
type Error = self::error::ConversionError;
fn try_from(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<&::std::string::String> for ResponseStatus {
type Error = self::error::ConversionError;
fn try_from(
value: &::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<::std::string::String> for ResponseStatus {
type Error = self::error::ConversionError;
fn try_from(
value: ::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
///What the agent may do: `receive` = read inbound on this conversation; `converse` = read and reply.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "What the agent may do: `receive` = read inbound on this conversation; `converse` = read and reply.",
/// "type": "string",
/// "enum": [
/// "receive",
/// "converse"
/// ]
///}
/// ```
/// </details>
#[derive(
::serde::Deserialize,
::serde::Serialize,
Clone,
Copy,
Debug,
Eq,
Hash,
Ord,
PartialEq,
PartialOrd,
)]
pub enum Scope {
#[serde(rename = "receive")]
Receive,
#[serde(rename = "converse")]
Converse,
}
impl ::std::fmt::Display for Scope {
fn fmt(&self, f: &mut ::std::fmt::Formatter<'_>) -> ::std::fmt::Result {
match *self {
Self::Receive => f.write_str("receive"),
Self::Converse => f.write_str("converse"),
}
}
}
impl ::std::str::FromStr for Scope {
type Err = self::error::ConversionError;
fn from_str(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
match value {
"receive" => Ok(Self::Receive),
"converse" => Ok(Self::Converse),
_ => Err("invalid value".into()),
}
}
}
impl ::std::convert::TryFrom<&str> for Scope {
type Error = self::error::ConversionError;
fn try_from(value: &str) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<&::std::string::String> for Scope {
type Error = self::error::ConversionError;
fn try_from(
value: &::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl ::std::convert::TryFrom<::std::string::String> for Scope {
type Error = self::error::ConversionError;
fn try_from(
value: ::std::string::String,
) -> ::std::result::Result<Self, self::error::ConversionError> {
value.parse()
}
}
impl crate::Payload for Payload {
const TYPE_URI: &'static str = "https://trusttasks.org/spec/consent/request/1.0";
const IS_PROOF_REQUIRED: bool = true;
const IS_RECIPIENT_REQUIRED: bool = true;
}
impl crate::Payload for Response {
const TYPE_URI: &'static str = "https://trusttasks.org/spec/consent/request/1.0#response";
const IS_PROOF_REQUIRED: bool = true;
const IS_RECIPIENT_REQUIRED: bool = true;
}
#[cfg(feature = "validate")]
impl crate::validate::ValidatedPayload for Payload {
const SCHEMA_JSON: &'static str = "{\n \"$defs\": {\n \"ConsentSubject\": {\n \"additionalProperties\": false,\n \"description\": \"The platform-agnostic identifier of WHAT is being consented to: one conversation, for one agent.\",\n \"properties\": {\n \"agent\": {\n \"description\": \"VID (DID) of the AI agent the conversation would reach.\",\n \"minLength\": 1,\n \"type\": \"string\"\n },\n \"conversationRef\": {\n \"description\": \"The bridge's OPAQUE conversation handle (e.g. \\\"sig-1a2b3c4d\\\"). NEVER the raw platform address — the VTA never learns the phone number / chat id.\",\n \"minLength\": 1,\n \"type\": \"string\"\n },\n \"kind\": {\n \"$ref\": \"#/$defs/Kind\"\n },\n \"platform\": {\n \"description\": \"Messaging-platform tag, e.g. \\\"signal\\\", \\\"whatsapp\\\", \\\"slack\\\".\",\n \"minLength\": 1,\n \"type\": \"string\"\n }\n },\n \"required\": [\n \"platform\",\n \"conversationRef\",\n \"kind\",\n \"agent\"\n ],\n \"type\": \"object\"\n },\n \"Ext\": {\n \"additionalProperties\": true,\n \"description\": \"Vendor-namespaced extension object per SPEC.md §4.5.1. Each immediate key MUST be a reverse-DNS namespace; structure under each namespace is opaque to the framework.\",\n \"minProperties\": 1,\n \"propertyNames\": {\n \"pattern\": \"^[a-z][a-z0-9-]*(\\\\.[a-z0-9-]+)+$\"\n },\n \"title\": \"Ext\",\n \"type\": \"object\"\n },\n \"Kind\": {\n \"description\": \"The interaction kind: a 1:1 direct message, a multi-party group, or a broadcast channel.\",\n \"enum\": [\n \"dm\",\n \"group\",\n \"channel\"\n ],\n \"type\": \"string\"\n },\n \"Response\": {\n \"$anchor\": \"response\",\n \"additionalProperties\": false,\n \"description\": \"Synchronous ack from the VTA: a prompt was routed (accepted) or the request was refused. The actual decision arrives out-of-band as a consent/decision.\",\n \"properties\": {\n \"ext\": {\n \"$ref\": \"#/$defs/Ext\"\n },\n \"reason\": {\n \"description\": \"Required when status is `refused`.\",\n \"type\": \"string\"\n },\n \"requestId\": {\n \"description\": \"The VTA's id for the pending consent (set when accepted), for correlation and polling.\",\n \"type\": \"string\"\n },\n \"status\": {\n \"description\": \"`accepted` = a pending consent was minted and an approval prompt routed. `refused` = not actionable; `reason` MUST be set.\",\n \"enum\": [\n \"accepted\",\n \"refused\"\n ],\n \"type\": \"string\"\n }\n },\n \"required\": [\n \"status\"\n ],\n \"title\": \"Consent Request — response payload\",\n \"type\": \"object\"\n },\n \"Scope\": {\n \"description\": \"What the agent may do: `receive` = read inbound on this conversation; `converse` = read and reply.\",\n \"enum\": [\n \"receive\",\n \"converse\"\n ],\n \"type\": \"string\"\n }\n },\n \"$id\": \"https://trusttasks.org/spec/consent/request/1.0\",\n \"$schema\": \"https://json-schema.org/draft/2020-12/schema\",\n \"additionalProperties\": false,\n \"description\": \"A bridge asks the VTA whether an inbound conversation may reach an AI agent, prompting operator consent on first contact.\",\n \"properties\": {\n \"challenge\": {\n \"description\": \"base64url-encoded nonce (≥128 bits entropy) echoed by the matching consent/decision, so the bridge can correlate the decision to this request.\",\n \"minLength\": 16,\n \"type\": \"string\"\n },\n \"contextHint\": {\n \"description\": \"Optional VTA context path the bridge runs under, to scope approver resolution.\",\n \"type\": \"string\"\n },\n \"displayHint\": {\n \"description\": \"Optional operator-facing label (e.g. \\\"Signal group 'Family'\\\"). MUST NOT contain a raw platform address.\",\n \"type\": \"string\"\n },\n \"ext\": {\n \"$ref\": \"#/$defs/Ext\"\n },\n \"firstMessageDigest\": {\n \"description\": \"Optional multihash digest of the held first message, binding the request to concrete content.\",\n \"type\": \"string\"\n },\n \"scope\": {\n \"$ref\": \"#/$defs/Scope\",\n \"description\": \"The access the bridge seeks for the agent on this conversation.\"\n },\n \"subject\": {\n \"$ref\": \"#/$defs/ConsentSubject\"\n }\n },\n \"required\": [\n \"subject\",\n \"scope\",\n \"challenge\"\n ],\n \"title\": \"Consent Request — payload\",\n \"type\": \"object\"\n}\n";
}
#[cfg(test)]
mod conformance {
//! Round-trip tests harvested from the spec's `spec.md`,
//! plus a `rejects_invalid_examples` test for any fixtures
//! in `payload.invalid-examples.json` (validate feature).
#[test]
fn response_example_1() {
const JSON: &str = "{\n \"id\": \"urn:uuid:consent-req-resp-0001\",\n \"type\": \"https://trusttasks.org/spec/consent/request/1.0#response\",\n \"threadId\": \"urn:uuid:consent-req-0001-0000-0000-000000000001\",\n \"issuer\": \"did:webvh:example:vta\",\n \"recipient\": \"did:webvh:example:bridge\",\n \"issuedAt\": \"2026-06-17T15:00:01Z\",\n \"payload\": {\n \"status\": \"accepted\",\n \"requestId\": \"consent-pending-7f3a\"\n }\n}\n";
let doc: crate::TrustTask<super::Response> =
serde_json::from_str(JSON).expect("deserialize response example");
let rendered = serde_json::to_value(&doc).expect("re-serialize");
let expected: serde_json::Value = serde_json::from_str(JSON).expect("re-parse expected");
assert_eq!(rendered, expected, "response example failed round-trip");
}
/// Each fixture in `payload.invalid-examples.json` MUST be
/// rejected by at least one of: serde deserialization, or
/// JSON-Schema validation under the `validate` feature. The
/// fixture file documents the producer-side bug class that
/// each payload exemplifies; this generated test pins it.
#[cfg(feature = "validate")]
#[test]
fn rejects_invalid_examples() {
use crate::validate::ValidatedPayload;
let fixtures: &[(&str, &str)] = &[
(
"Missing required subject.",
"{\n \"challenge\": \"Q29uc2VudENoYWxsZW5nZU5vbmNlWFla\",\n \"scope\": \"converse\"\n}",
),
(
"Subject missing required agent.",
"{\n \"challenge\": \"Q29uc2VudENoYWxsZW5nZU5vbmNlWFla\",\n \"scope\": \"converse\",\n \"subject\": {\n \"conversationRef\": \"sig-1a2b3c4d\",\n \"kind\": \"group\",\n \"platform\": \"signal\"\n }\n}",
),
(
"Invalid kind.",
"{\n \"challenge\": \"Q29uc2VudENoYWxsZW5nZU5vbmNlWFla\",\n \"scope\": \"converse\",\n \"subject\": {\n \"agent\": \"did:key:z6MkAgentExample\",\n \"conversationRef\": \"sig-1a2b3c4d\",\n \"kind\": \"broadcast\",\n \"platform\": \"signal\"\n }\n}",
),
(
"Challenge too short.",
"{\n \"challenge\": \"short\",\n \"scope\": \"converse\",\n \"subject\": {\n \"agent\": \"did:key:z6MkAgentExample\",\n \"conversationRef\": \"sig-1a2b3c4d\",\n \"kind\": \"group\",\n \"platform\": \"signal\"\n }\n}",
),
(
"Unknown property.",
"{\n \"challenge\": \"Q29uc2VudENoYWxsZW5nZU5vbmNlWFla\",\n \"rawNumber\": \"+15551234567\",\n \"scope\": \"converse\",\n \"subject\": {\n \"agent\": \"did:key:z6MkAgentExample\",\n \"conversationRef\": \"sig-1a2b3c4d\",\n \"kind\": \"group\",\n \"platform\": \"signal\"\n }\n}",
),
];
for (i, (note, raw)) in fixtures.iter().enumerate() {
let value: serde_json::Value = match serde_json::from_str(raw) {
Ok(v) => v,
Err(_) => continue,
};
let serde_ok = serde_json::from_value::<super::Payload>(value.clone()).is_ok();
let schema_ok = super::Payload::validate_value(&value).is_ok();
assert!(
!(serde_ok && schema_ok),
"invalid-example #{} ({:?}) was accepted by both serde and JSON Schema; \
the fixture's stated failure class is no longer caught:\n{}",
i + 1,
note,
raw
);
}
}
}