//! Generated by `trust-tasks-codegen` — do not edit by hand.
//!
//! Spec slug: `acl/list`. Version: `0.1`.
#[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())
}
}
}
///`AclEntry`
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "type": "object",
/// "required": [
/// "role",
/// "subject"
/// ],
/// "properties": {
/// "createdAt": {
/// "type": "string",
/// "format": "date-time"
/// },
/// "createdBy": {
/// "type": "string"
/// },
/// "expiresAt": {
/// "type": "string",
/// "format": "date-time"
/// },
/// "label": {
/// "type": "string"
/// },
/// "metadata": {
/// "type": "object"
/// },
/// "role": {
/// "type": "string"
/// },
/// "scopes": {
/// "type": "array",
/// "items": {
/// "type": "string"
/// }
/// },
/// "subject": {
/// "type": "string"
/// },
/// "updatedAt": {
/// "type": "string",
/// "format": "date-time"
/// },
/// "updatedBy": {
/// "type": "string"
/// }
/// },
/// "additionalProperties": false
///}
/// ```
/// </details>
#[derive(::serde::Deserialize, ::serde::Serialize, Clone, Debug)]
#[serde(deny_unknown_fields)]
pub struct AclEntry {
#[serde(
rename = "createdAt",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub created_at: ::std::option::Option<::chrono::DateTime<::chrono::offset::Utc>>,
#[serde(
rename = "createdBy",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub created_by: ::std::option::Option<::std::string::String>,
#[serde(
rename = "expiresAt",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub expires_at: ::std::option::Option<::chrono::DateTime<::chrono::offset::Utc>>,
#[serde(default, skip_serializing_if = "::std::option::Option::is_none")]
pub label: ::std::option::Option<::std::string::String>,
#[serde(default, skip_serializing_if = "::serde_json::Map::is_empty")]
pub metadata: ::serde_json::Map<::std::string::String, ::serde_json::Value>,
pub role: ::std::string::String,
#[serde(default, skip_serializing_if = "::std::vec::Vec::is_empty")]
pub scopes: ::std::vec::Vec<::std::string::String>,
pub subject: ::std::string::String,
#[serde(
rename = "updatedAt",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub updated_at: ::std::option::Option<::chrono::DateTime<::chrono::offset::Utc>>,
#[serde(
rename = "updatedBy",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub updated_by: ::std::option::Option<::std::string::String>,
}
impl ::std::convert::From<&AclEntry> for AclEntry {
fn from(value: &AclEntry) -> Self {
value.clone()
}
}
///`Payload`
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "$id": "https://trusttasks.org/spec/acl/list/0.1",
/// "title": "Payload",
/// "type": "object",
/// "properties": {
/// "cursor": {
/// "description": "Opaque continuation token returned by the maintainer in a previous response.",
/// "type": "string"
/// },
/// "pageSize": {
/// "description": "Maximum number of entries to return. Maintainer-defined default and ceiling.",
/// "type": "integer",
/// "maximum": 1000.0,
/// "minimum": 1.0
/// },
/// "role": {
/// "description": "Optional filter — only entries with this role are returned.",
/// "type": "string",
/// "minLength": 1
/// },
/// "scope": {
/// "description": "Optional filter — only entries whose scopes include this string are returned.",
/// "type": "string",
/// "minLength": 1
/// },
/// "subjectPrefix": {
/// "description": "Optional filter — only entries whose subject VID starts with this prefix are returned.",
/// "type": "string",
/// "minLength": 1
/// }
/// },
/// "additionalProperties": false
///}
/// ```
/// </details>
#[derive(::serde::Deserialize, ::serde::Serialize, Clone, Debug)]
#[serde(deny_unknown_fields)]
pub struct Payload {
///Opaque continuation token returned by the maintainer in a previous response.
#[serde(default, skip_serializing_if = "::std::option::Option::is_none")]
pub cursor: ::std::option::Option<::std::string::String>,
///Maximum number of entries to return. Maintainer-defined default and ceiling.
#[serde(
rename = "pageSize",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub page_size: ::std::option::Option<::std::num::NonZeroU64>,
///Optional filter — only entries with this role are returned.
#[serde(default, skip_serializing_if = "::std::option::Option::is_none")]
pub role: ::std::option::Option<PayloadRole>,
///Optional filter — only entries whose scopes include this string are returned.
#[serde(default, skip_serializing_if = "::std::option::Option::is_none")]
pub scope: ::std::option::Option<PayloadScope>,
///Optional filter — only entries whose subject VID starts with this prefix are returned.
#[serde(
rename = "subjectPrefix",
default,
skip_serializing_if = "::std::option::Option::is_none"
)]
pub subject_prefix: ::std::option::Option<PayloadSubjectPrefix>,
}
impl ::std::convert::From<&Payload> for Payload {
fn from(value: &Payload) -> Self {
value.clone()
}
}
impl ::std::default::Default for Payload {
fn default() -> Self {
Self {
cursor: Default::default(),
page_size: Default::default(),
role: Default::default(),
scope: Default::default(),
subject_prefix: Default::default(),
}
}
}
///Optional filter — only entries with this role are returned.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "Optional filter — only entries with this role are returned.",
/// "type": "string",
/// "minLength": 1
///}
/// ```
/// </details>
#[derive(::serde::Serialize, Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[serde(transparent)]
pub struct PayloadRole(::std::string::String);
impl ::std::ops::Deref for PayloadRole {
type Target = ::std::string::String;
fn deref(&self) -> &::std::string::String {
&self.0
}
}
impl ::std::convert::From<PayloadRole> for ::std::string::String {
fn from(value: PayloadRole) -> Self {
value.0
}
}
impl ::std::convert::From<&PayloadRole> for PayloadRole {
fn from(value: &PayloadRole) -> Self {
value.clone()
}
}
impl ::std::str::FromStr for PayloadRole {
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 PayloadRole {
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 PayloadRole {
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 PayloadRole {
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 PayloadRole {
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())
})
}
}
///Optional filter — only entries whose scopes include this string are returned.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "Optional filter — only entries whose scopes include this string are returned.",
/// "type": "string",
/// "minLength": 1
///}
/// ```
/// </details>
#[derive(::serde::Serialize, Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[serde(transparent)]
pub struct PayloadScope(::std::string::String);
impl ::std::ops::Deref for PayloadScope {
type Target = ::std::string::String;
fn deref(&self) -> &::std::string::String {
&self.0
}
}
impl ::std::convert::From<PayloadScope> for ::std::string::String {
fn from(value: PayloadScope) -> Self {
value.0
}
}
impl ::std::convert::From<&PayloadScope> for PayloadScope {
fn from(value: &PayloadScope) -> Self {
value.clone()
}
}
impl ::std::str::FromStr for PayloadScope {
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 PayloadScope {
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 PayloadScope {
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 PayloadScope {
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 PayloadScope {
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())
})
}
}
///Optional filter — only entries whose subject VID starts with this prefix are returned.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "description": "Optional filter — only entries whose subject VID starts with this prefix are returned.",
/// "type": "string",
/// "minLength": 1
///}
/// ```
/// </details>
#[derive(::serde::Serialize, Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[serde(transparent)]
pub struct PayloadSubjectPrefix(::std::string::String);
impl ::std::ops::Deref for PayloadSubjectPrefix {
type Target = ::std::string::String;
fn deref(&self) -> &::std::string::String {
&self.0
}
}
impl ::std::convert::From<PayloadSubjectPrefix> for ::std::string::String {
fn from(value: PayloadSubjectPrefix) -> Self {
value.0
}
}
impl ::std::convert::From<&PayloadSubjectPrefix> for PayloadSubjectPrefix {
fn from(value: &PayloadSubjectPrefix) -> Self {
value.clone()
}
}
impl ::std::str::FromStr for PayloadSubjectPrefix {
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 PayloadSubjectPrefix {
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 PayloadSubjectPrefix {
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 PayloadSubjectPrefix {
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 PayloadSubjectPrefix {
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 success response to an acl/list request. Carried in a Trust Task document whose type is https://trusttasks.org/spec/acl/list/0.1#response.
///
/// <details><summary>JSON schema</summary>
///
/// ```json
///{
/// "title": "Response",
/// "description": "The success response to an acl/list request. Carried in a Trust Task document whose type is https://trusttasks.org/spec/acl/list/0.1#response.",
/// "type": "object",
/// "required": [
/// "entries",
/// "truncated"
/// ],
/// "properties": {
/// "cursor": {
/// "description": "Opaque continuation token to fetch the next page. Present only when `truncated` is true AND the maintainer supports pagination from this point. Consumers MUST treat the cursor as opaque and re-send it verbatim.",
/// "type": "string"
/// },
/// "entries": {
/// "description": "Matching AclEntry items, in maintainer-defined order. May be empty.",
/// "type": "array",
/// "items": {
/// "$ref": "#/definitions/AclEntry"
/// }
/// },
/// "redactedFields": {
/// "description": "Names of AclEntry fields the maintainer redacted from every returned entry (for example, ['metadata']).",
/// "type": "array",
/// "items": {
/// "type": "string"
/// }
/// },
/// "truncated": {
/// "description": "true when more matching entries exist beyond `entries`; false when this response is the complete result. Independent of `cursor`: a maintainer MAY truncate without supporting pagination, in which case `truncated` is true and `cursor` is absent.",
/// "type": "boolean"
/// }
/// },
/// "additionalProperties": false,
/// "$anchor": "response"
///}
/// ```
/// </details>
#[derive(::serde::Deserialize, ::serde::Serialize, Clone, Debug)]
#[serde(deny_unknown_fields)]
pub struct Response {
///Opaque continuation token to fetch the next page. Present only when `truncated` is true AND the maintainer supports pagination from this point. Consumers MUST treat the cursor as opaque and re-send it verbatim.
#[serde(default, skip_serializing_if = "::std::option::Option::is_none")]
pub cursor: ::std::option::Option<::std::string::String>,
///Matching AclEntry items, in maintainer-defined order. May be empty.
pub entries: ::std::vec::Vec<AclEntry>,
///Names of AclEntry fields the maintainer redacted from every returned entry (for example, ['metadata']).
#[serde(
rename = "redactedFields",
default,
skip_serializing_if = "::std::vec::Vec::is_empty"
)]
pub redacted_fields: ::std::vec::Vec<::std::string::String>,
///true when more matching entries exist beyond `entries`; false when this response is the complete result. Independent of `cursor`: a maintainer MAY truncate without supporting pagination, in which case `truncated` is true and `cursor` is absent.
pub truncated: bool,
}
impl ::std::convert::From<&Response> for Response {
fn from(value: &Response) -> Self {
value.clone()
}
}
impl crate::Payload for Payload {
const TYPE_URI: &'static str = "https://trusttasks.org/spec/acl/list/0.1";
}
impl crate::Payload for Response {
const TYPE_URI: &'static str = "https://trusttasks.org/spec/acl/list/0.1#response";
}
#[cfg(feature = "validate")]
impl crate::validate::ValidatedPayload for Payload {
const SCHEMA_JSON: &'static str = "{\n \"$schema\": \"https://json-schema.org/draft/2020-12/schema\",\n \"$id\": \"https://trusttasks.org/spec/acl/list/0.1\",\n \"title\": \"ACL List — payload\",\n \"type\": \"object\",\n \"additionalProperties\": false,\n \"properties\": {\n \"role\": {\n \"type\": \"string\",\n \"minLength\": 1,\n \"description\": \"Optional filter — only entries with this role are returned.\"\n },\n \"scope\": {\n \"type\": \"string\",\n \"minLength\": 1,\n \"description\": \"Optional filter — only entries whose scopes include this string are returned.\"\n },\n \"subjectPrefix\": {\n \"type\": \"string\",\n \"minLength\": 1,\n \"description\": \"Optional filter — only entries whose subject VID starts with this prefix are returned.\"\n },\n \"pageSize\": {\n \"type\": \"integer\",\n \"minimum\": 1,\n \"maximum\": 1000,\n \"description\": \"Maximum number of entries to return. Maintainer-defined default and ceiling.\"\n },\n \"cursor\": {\n \"type\": \"string\",\n \"description\": \"Opaque continuation token returned by the maintainer in a previous response.\"\n }\n },\n \"$defs\": {\n \"AclEntry\": {\n \"type\": \"object\",\n \"additionalProperties\": false,\n \"required\": [\"subject\", \"role\"],\n \"properties\": {\n \"subject\": { \"type\": \"string\" },\n \"role\": { \"type\": \"string\" },\n \"scopes\": { \"type\": \"array\", \"items\": { \"type\": \"string\" } },\n \"label\": { \"type\": \"string\" },\n \"createdAt\": { \"type\": \"string\", \"format\": \"date-time\" },\n \"createdBy\": { \"type\": \"string\" },\n \"updatedAt\": { \"type\": \"string\", \"format\": \"date-time\" },\n \"updatedBy\": { \"type\": \"string\" },\n \"expiresAt\": { \"type\": \"string\", \"format\": \"date-time\" },\n \"metadata\": { \"type\": \"object\" }\n }\n },\n \"Response\": {\n \"$anchor\": \"response\",\n \"title\": \"ACL List — response payload\",\n \"description\": \"The success response to an acl/list request. Carried in a Trust Task document whose type is https://trusttasks.org/spec/acl/list/0.1#response.\",\n \"type\": \"object\",\n \"additionalProperties\": false,\n \"required\": [\"entries\", \"truncated\"],\n \"properties\": {\n \"entries\": {\n \"type\": \"array\",\n \"items\": { \"$ref\": \"#/$defs/AclEntry\" },\n \"description\": \"Matching AclEntry items, in maintainer-defined order. May be empty.\"\n },\n \"truncated\": {\n \"type\": \"boolean\",\n \"description\": \"true when more matching entries exist beyond `entries`; false when this response is the complete result. Independent of `cursor`: a maintainer MAY truncate without supporting pagination, in which case `truncated` is true and `cursor` is absent.\"\n },\n \"cursor\": {\n \"type\": \"string\",\n \"description\": \"Opaque continuation token to fetch the next page. Present only when `truncated` is true AND the maintainer supports pagination from this point. Consumers MUST treat the cursor as opaque and re-send it verbatim.\"\n },\n \"redactedFields\": {\n \"type\": \"array\",\n \"items\": { \"type\": \"string\" },\n \"description\": \"Names of AclEntry fields the maintainer redacted from every returned entry (for example, ['metadata']).\"\n }\n }\n }\n }\n}\n";
}
#[cfg(test)]
mod conformance {
//! Round-trip tests harvested from the spec's `spec.md`.
#[test]
fn request_example_1() {
const JSON: &str = "{\n \"id\": \"2e2a1c44-7b81-4d3e-9b51-7a3c89e3d1f2\",\n \"type\": \"https://trusttasks.org/spec/acl/list/0.1\",\n \"issuer\": \"did:web:admin.example\",\n \"recipient\": \"did:web:maintainer.example\",\n \"issuedAt\": \"2026-06-15T10:00:00Z\",\n \"payload\": {}\n}\n";
let doc: crate::TrustTask<super::Payload> =
serde_json::from_str(JSON).expect("deserialize request 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, "request example failed round-trip");
}
#[test]
fn request_example_2() {
const JSON: &str = "{\n \"id\": \"5b3c5e2a-1b81-4d3e-9b51-7a3c89e3d1f2\",\n \"type\": \"https://trusttasks.org/spec/acl/list/0.1\",\n \"issuer\": \"did:web:auditor.example\",\n \"recipient\": \"did:web:maintainer.example\",\n \"issuedAt\": \"2026-06-15T10:05:00Z\",\n \"payload\": {\n \"role\": \"admin\",\n \"pageSize\": 50\n }\n}\n";
let doc: crate::TrustTask<super::Payload> =
serde_json::from_str(JSON).expect("deserialize request 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, "request example failed round-trip");
}
#[test]
fn request_example_3() {
const JSON: &str = "{\n \"id\": \"7e2c5e2a-1b81-4d3e-9b51-7a3c89e3d1f2\",\n \"type\": \"https://trusttasks.org/spec/acl/list/0.1\",\n \"issuer\": \"did:web:auditor.example\",\n \"recipient\": \"did:web:maintainer.example\",\n \"issuedAt\": \"2026-06-15T10:06:00Z\",\n \"payload\": {\n \"role\": \"admin\",\n \"pageSize\": 50,\n \"cursor\": \"eyJvZmZzZXQiOjUwfQ\"\n }\n}\n";
let doc: crate::TrustTask<super::Payload> =
serde_json::from_str(JSON).expect("deserialize request 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, "request example failed round-trip");
}
#[test]
fn response_example_1() {
const JSON: &str = "{\n \"id\": \"6c3c5e2a-1b81-4d3e-9b51-7a3c89e3d1f3\",\n \"type\": \"https://trusttasks.org/spec/acl/list/0.1#response\",\n \"threadId\": \"5b3c5e2a-1b81-4d3e-9b51-7a3c89e3d1f2\",\n \"issuer\": \"did:web:maintainer.example\",\n \"recipient\": \"did:web:auditor.example\",\n \"issuedAt\": \"2026-06-15T10:05:01Z\",\n \"payload\": {\n \"entries\": [\n {\n \"subject\": \"did:web:alice.example\",\n \"role\": \"admin\",\n \"label\": \"Alice — primary admin\",\n \"createdAt\": \"2026-05-16T10:00:00Z\",\n \"createdBy\": \"did:web:org.example\"\n },\n {\n \"subject\": \"did:web:carol.example\",\n \"role\": \"admin\",\n \"createdAt\": \"2026-05-18T08:30:00Z\",\n \"createdBy\": \"did:web:alice.example\"\n }\n ],\n \"truncated\": true,\n \"cursor\": \"eyJvZmZzZXQiOjUwfQ\"\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");
}
#[test]
fn response_example_2() {
const JSON: &str = "{\n \"id\": \"7e2c5e2a-1b81-4d3e-9b51-7a3c89e3d1f3\",\n \"type\": \"https://trusttasks.org/spec/acl/list/0.1#response\",\n \"threadId\": \"5b3c5e2a-1b81-4d3e-9b51-7a3c89e3d1f2\",\n \"issuer\": \"did:web:maintainer.example\",\n \"recipient\": \"did:web:auditor.example\",\n \"issuedAt\": \"2026-06-15T10:06:00Z\",\n \"payload\": {\n \"entries\": [\n {\n \"subject\": \"did:web:alice.example\",\n \"role\": \"admin\",\n \"createdAt\": \"2026-05-16T10:00:00Z\",\n \"createdBy\": \"did:web:org.example\"\n }\n ],\n \"truncated\": false,\n \"redactedFields\": [\"metadata\", \"label\"]\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");
}
}