Skip to main content

vta_cli_common/commands/
contexts.rs

1use std::io::{self, Write};
2
3use ratatui::{
4    layout::Constraint,
5    style::{Color, Modifier, Style},
6    widgets::{Block, Cell, Row, Table},
7};
8use vta_sdk::client::{ContextResponse, CreateDidWebvhRequest, UpdateContextRequest};
9use vta_sdk::context_provision::{ContextProvisionBundle, ProvisionedDid};
10use vta_sdk::prelude::*;
11use vta_sdk::sealed_transfer::SealedPayloadV1;
12
13use crate::render::{is_full_display, print_full_entry, print_full_list_title, print_widget};
14use crate::sealed_producer::{SealedRecipient, seal_for_recipient};
15
16pub struct ProvisionDidOptions {
17    pub server_id: Option<String>,
18    pub did_url: Option<String>,
19    pub portable: bool,
20    pub add_mediator_service: bool,
21    pub pre_rotation_count: u32,
22}
23
24pub async fn cmd_context_bootstrap(
25    client: &VtaClient,
26    id: &str,
27    name: &str,
28    description: Option<String>,
29    admin_label: Option<String>,
30    recipient: SealedRecipient,
31) -> Result<(), Box<dyn std::error::Error>> {
32    let mut ctx_req = CreateContextRequest::new(id, name);
33    if let Some(desc) = description {
34        ctx_req = ctx_req.description(desc);
35    }
36    let ctx = client.create_context(ctx_req).await?;
37    println!("Context created:");
38    println!("  ID:        {}", ctx.id);
39    println!("  Name:      {}", ctx.name);
40    println!("  Base Path: {}", ctx.base_path);
41
42    // Fetch VTA config so the minted CredentialBundle carries VTA DID/URL.
43    let config = client.get_config().await?;
44    let vta_did = config
45        .community_vta_did
46        .clone()
47        .ok_or("VTA DID not configured — cannot mint admin credential")?;
48    let vta_url = config.public_url.clone();
49
50    // Mint admin did:key locally + register ACL. The VTA never sees the
51    // private half; the full bundle reaches the recipient via sealed transfer.
52    let (admin_bundle, admin_did) = crate::local_keygen::generate_admin_did_key(vta_did, vta_url);
53    let mut acl_req =
54        vta_sdk::client::CreateAclRequest::new(&admin_did, "admin").contexts(vec![id.to_string()]);
55    if let Some(l) = admin_label {
56        acl_req = acl_req.label(l);
57    }
58    client.create_acl(acl_req).await?;
59
60    let sealed = seal_for_recipient(
61        &recipient,
62        &SealedPayloadV1::AdminCredential(Box::new(admin_bundle)),
63    )
64    .await?;
65    println!();
66    println!("Admin credential created:");
67    println!("  DID:  {admin_did}");
68    println!("  Role: admin");
69    if let Some(ref label) = recipient.label {
70        println!("  Recipient: {label}");
71    }
72    println!();
73
74    crate::sealed_producer::emit_sealed_output(&sealed, None)?;
75    Ok(())
76}
77
78/// Render a list of context records — table view by default, full
79/// `key: value` blocks when `--full-display` is set.
80///
81/// Shared by the online (`pnm contexts list`, REST) and offline
82/// (`vta contexts list`, keystore-direct) paths so both render
83/// identically.
84pub fn render_context_list(contexts: &[ContextResponse]) {
85    if contexts.is_empty() {
86        println!("No contexts found.");
87        return;
88    }
89
90    if is_full_display() {
91        print_full_list_title("Contexts", contexts.len());
92        for ctx in contexts {
93            let did = ctx.did.as_deref().unwrap_or("—");
94            let created = ctx
95                .created_at
96                .with_timezone(&chrono::Local)
97                .format("%Y-%m-%d %H:%M:%S %:z")
98                .to_string();
99            print_full_entry(&[
100                ("ID", &ctx.id),
101                ("Name", &ctx.name),
102                ("DID", did),
103                ("Base Path", &ctx.base_path),
104                ("Created", &created),
105            ]);
106        }
107        return;
108    }
109
110    let header_style = Style::default()
111        .fg(Color::White)
112        .add_modifier(Modifier::BOLD);
113    let header = Row::new(vec!["ID", "Name", "DID", "Base Path", "Created"])
114        .style(header_style)
115        .bottom_margin(1);
116
117    let rows: Vec<Row> = contexts
118        .iter()
119        .map(|ctx| {
120            let did = ctx.did.clone().unwrap_or_else(|| "\u{2014}".into());
121            let created = ctx
122                .created_at
123                .with_timezone(&chrono::Local)
124                .format("%Y-%m-%d")
125                .to_string();
126
127            Row::new(vec![
128                Cell::from(ctx.id.clone()),
129                Cell::from(ctx.name.clone()),
130                Cell::from(did).style(Style::default().fg(Color::DarkGray)),
131                Cell::from(ctx.base_path.clone()),
132                Cell::from(created),
133            ])
134        })
135        .collect();
136
137    let title = format!(" Contexts ({}) ", contexts.len());
138
139    // DID field carries full did:webvh / did:key values (40+ chars).
140    // Use `Min` so it expands on wide terminals rather than truncating
141    // at the former fixed 30-char width.
142    let table = Table::new(
143        rows,
144        [
145            Constraint::Min(16),    // ID
146            Constraint::Min(20),    // Name
147            Constraint::Min(40),    // DID
148            Constraint::Length(16), // Base Path
149            Constraint::Length(10), // Created
150        ],
151    )
152    .header(header)
153    .column_spacing(2)
154    .block(
155        Block::bordered()
156            .title(title)
157            .border_style(Style::default().fg(Color::DarkGray)),
158    );
159
160    let height = contexts.len() as u16 + 4;
161    print_widget(table, height);
162}
163
164/// Render a single context record's details, used by `get` and the
165/// success path of `update`. Shared by online + offline call sites.
166pub fn render_context_record(ctx: &ContextResponse) {
167    println!("ID:          {}", ctx.id);
168    println!("Name:        {}", ctx.name);
169    println!("DID:         {}", ctx.did.as_deref().unwrap_or("(not set)"));
170    println!(
171        "Description: {}",
172        ctx.description.as_deref().unwrap_or("(not set)")
173    );
174    println!("Base Path:   {}", ctx.base_path);
175    println!(
176        "Created At:  {}",
177        crate::duration::format_local_datetime(ctx.created_at)
178    );
179    println!(
180        "Updated At:  {}",
181        crate::duration::format_local_datetime(ctx.updated_at)
182    );
183}
184
185pub async fn cmd_context_list(client: &VtaClient) -> Result<(), Box<dyn std::error::Error>> {
186    let resp = client.list_contexts().await?;
187    if crate::render::is_json_output() {
188        crate::render::print_json(&resp.contexts)?;
189        return Ok(());
190    }
191    render_context_list(&resp.contexts);
192    Ok(())
193}
194
195pub async fn cmd_context_get(
196    client: &VtaClient,
197    id: &str,
198) -> Result<(), Box<dyn std::error::Error>> {
199    let resp = client.get_context(id).await?;
200    render_context_record(&resp);
201    Ok(())
202}
203
204/// Options for optionally creating a context-scoped admin ACL entry as
205/// part of `pnm contexts create`.
206///
207/// Omit the entire struct to create the context without touching the ACL —
208/// the historical behaviour. Supply a [`AdminAclOptions::did`] to atomically
209/// create an `admin`-role ACL entry scoped to the new context in the same
210/// CLI invocation.
211///
212/// Setting [`AdminAclOptions::expires_at`] flips the entry from **permanent**
213/// to a **setup ACL** that auto-expires (pruned by the VTA's ACL sweeper) if
214/// the admin never authenticates and rotates to a fresh did:key.
215#[derive(Debug, Default, Clone)]
216pub struct AdminAclOptions {
217    /// DID to grant admin access to. Must start with `did:`.
218    pub did: Option<String>,
219    /// Human-readable label stored on the ACL entry.
220    pub label: Option<String>,
221    /// Unix-epoch seconds at which the entry auto-expires. `None` = permanent.
222    pub expires_at: Option<u64>,
223    /// Raw `--admin-expires` input (e.g. `"1h"`). Preserved alongside the
224    /// resolved `expires_at` so conflict hints can re-emit the operator's
225    /// original duration verbatim instead of a drift-skewed seconds value.
226    pub expires_duration: Option<String>,
227}
228
229impl AdminAclOptions {
230    fn is_requested(&self) -> bool {
231        self.did.is_some()
232    }
233}
234
235pub async fn cmd_context_create(
236    client: &VtaClient,
237    id: &str,
238    name: &str,
239    description: Option<String>,
240    admin: AdminAclOptions,
241) -> Result<(), Box<dyn std::error::Error>> {
242    use crate::render::{RESET, YELLOW};
243    use vta_sdk::error::VtaError;
244
245    let req = CreateContextRequest {
246        id: id.to_string(),
247        name: name.to_string(),
248        description,
249    };
250    let resp = match client.create_context(req).await {
251        Ok(r) => r,
252        // Friendly path when the operator's real intent was "grant this DID
253        // admin access to this context": the context already exists, so the
254        // scaffolding command is the wrong tool — point them at the ACL
255        // command and exit cleanly so repeated provisioning scripts don't
256        // choke on second runs.
257        Err(VtaError::Conflict(_)) if admin.is_requested() => {
258            let did = admin.did.as_deref().unwrap_or_default();
259            let bin = crate::render::bin_name();
260            eprintln!(
261                "{YELLOW}\u{26a0}{RESET}  Context '{id}' already exists — skipping context creation."
262            );
263            eprintln!();
264            eprintln!("  The --admin-did was NOT added. To grant admin access to an existing");
265            eprintln!("  context, use the ACL command directly:");
266            eprintln!();
267            let mut hint = format!("    {bin} acl create --did {did} --role admin --contexts {id}");
268            if let Some(label) = admin.label.as_deref() {
269                hint.push_str(&format!(" --label '{label}'"));
270            }
271            match (admin.expires_duration.as_deref(), admin.expires_at) {
272                // Prefer the raw duration the user typed — any latency between
273                // --admin-expires being parsed and the conflict firing would
274                // otherwise drift the re-rendered seconds (e.g. `1h` → `3599s`).
275                (Some(raw), _) => hint.push_str(&format!(" --expires {raw}")),
276                (None, Some(expires_at)) => {
277                    let remaining = expires_at.saturating_sub(crate::duration::now_unix());
278                    hint.push_str(&format!(" --expires {remaining}s"));
279                }
280                (None, None) => {}
281            }
282            eprintln!("{hint}");
283            return Ok(());
284        }
285        Err(e) => return Err(e.into()),
286    };
287    println!("Context created:");
288    println!("  ID:        {}", resp.id);
289    println!("  Name:      {}", resp.name);
290    println!("  Base Path: {}", resp.base_path);
291
292    if admin.is_requested() {
293        let did = admin.did.as_deref().unwrap_or_default();
294        if !did.starts_with("did:") {
295            return Err(format!(
296                "--admin-did must start with `did:` (got {did:?}) — context was created but no ACL entry was added"
297            )
298            .into());
299        }
300        let mut acl_req =
301            vta_sdk::client::CreateAclRequest::new(did, "admin").contexts(vec![id.to_string()]);
302        if let Some(label) = admin.label.as_deref() {
303            acl_req = acl_req.label(label);
304        }
305        if let Some(expires_at) = admin.expires_at {
306            acl_req = acl_req.expires_at(expires_at);
307        }
308        let acl = client.create_acl(acl_req).await?;
309
310        println!();
311        println!("Admin ACL entry created:");
312        println!("  DID:        {}", acl.did);
313        println!("  Role:       {}", acl.role);
314        println!("  Contexts:   {}", acl.allowed_contexts.join(", "));
315        if let Some(ref label) = acl.label {
316            println!("  Label:      {label}");
317        }
318        match acl.expires_at {
319            Some(secs) => {
320                println!(
321                    "  Expires at: {} ({}) — setup ACL",
322                    crate::duration::format_local_time(secs),
323                    crate::duration::format_remaining(secs),
324                );
325                println!();
326                println!("  The admin should authenticate before expiry. On first successful");
327                println!("  connect PNM rotates to a fresh long-lived did:key and replaces this");
328                println!("  temporary entry with a permanent one.");
329            }
330            None => println!("  Expires at: (permanent)"),
331        }
332    }
333
334    Ok(())
335}
336
337pub async fn cmd_context_update(
338    client: &VtaClient,
339    id: &str,
340    name: Option<String>,
341    did: Option<String>,
342    description: Option<String>,
343) -> Result<(), Box<dyn std::error::Error>> {
344    let req = UpdateContextRequest {
345        name,
346        did,
347        description,
348    };
349    let resp = client.update_context(id, req).await?;
350    println!("Context updated:");
351    render_context_record(&resp);
352    Ok(())
353}
354
355pub async fn cmd_context_update_did(
356    client: &VtaClient,
357    id: &str,
358    did: &str,
359) -> Result<(), Box<dyn std::error::Error>> {
360    let resp = client.update_context_did(id, did).await?;
361    println!("Context DID updated:");
362    println!("  ID:         {}", resp.id);
363    println!(
364        "  DID:        {}",
365        resp.did.as_deref().unwrap_or("(not set)")
366    );
367    println!(
368        "  Updated At: {}",
369        crate::duration::format_local_datetime(resp.updated_at)
370    );
371    Ok(())
372}
373
374/// Print the human-readable resource preview shown before context
375/// deletion. Returns `true` when the preview lists any resources
376/// (i.e. the caller should prompt for confirmation unless `--force`).
377///
378/// Shared by online + offline delete paths so both warn about exactly
379/// the same resource classes.
380pub fn render_delete_context_preview(
381    id: &str,
382    preview: &vta_sdk::protocols::context_management::delete::DeleteContextPreviewResultBody,
383) -> bool {
384    let has_resources = !preview.keys.is_empty()
385        || !preview.webvh_dids.is_empty()
386        || !preview.acl_entries_removed.is_empty()
387        || !preview.acl_entries_updated.is_empty();
388
389    if !has_resources {
390        return false;
391    }
392
393    println!(
394        "Deleting context '{}' will remove the following resources:\n",
395        id
396    );
397
398    if !preview.keys.is_empty() {
399        println!("  Keys ({}):", preview.keys.len());
400        for key in &preview.keys {
401            println!("    - {key}");
402        }
403    }
404
405    if !preview.webvh_dids.is_empty() {
406        println!("  WebVH DIDs ({}):", preview.webvh_dids.len());
407        for did in &preview.webvh_dids {
408            println!("    - {did}");
409        }
410    }
411
412    if !preview.acl_entries_removed.is_empty() {
413        println!(
414            "  ACL entries removed ({}):",
415            preview.acl_entries_removed.len()
416        );
417        for did in &preview.acl_entries_removed {
418            println!("    - {did}");
419        }
420    }
421
422    if !preview.acl_entries_updated.is_empty() {
423        println!(
424            "  ACL entries updated (context removed from access list) ({}):",
425            preview.acl_entries_updated.len()
426        );
427        for did in &preview.acl_entries_updated {
428            println!("    - {did}");
429        }
430    }
431
432    println!();
433    true
434}
435
436/// Read a `[y/N]` reply from stdin. Returns `true` for `y` / `yes`,
437/// `false` otherwise. Shared by the destructive shared commands so
438/// the prompt wording stays consistent.
439pub fn confirm_destructive(prompt: &str) -> Result<bool, Box<dyn std::error::Error>> {
440    print!("{prompt} [y/N] ");
441    io::stdout().flush()?;
442    let mut input = String::new();
443    io::stdin().read_line(&mut input)?;
444    let input = input.trim().to_lowercase();
445    Ok(input == "y" || input == "yes")
446}
447
448pub async fn cmd_context_delete(
449    client: &VtaClient,
450    id: &str,
451    force: bool,
452) -> Result<(), Box<dyn std::error::Error>> {
453    // Fetch a preview of what will be removed
454    let preview = client.preview_delete_context(id).await?;
455
456    let has_resources = render_delete_context_preview(id, &preview);
457
458    if has_resources && !force && !confirm_destructive("Proceed with deletion?")? {
459        println!("Aborted.");
460        return Ok(());
461    }
462
463    client.delete_context(id, true).await?;
464    println!("Context deleted: {id}");
465    Ok(())
466}
467
468pub async fn cmd_context_provision(
469    client: &VtaClient,
470    id: &str,
471    name: &str,
472    description: Option<String>,
473    admin_label: Option<String>,
474    did_opts: Option<ProvisionDidOptions>,
475    recipient: SealedRecipient,
476) -> Result<(), Box<dyn std::error::Error>> {
477    // 1. Create the context
478    eprintln!("Creating context '{id}'...");
479    let mut ctx_req = CreateContextRequest::new(id, name);
480    if let Some(desc) = description {
481        ctx_req = ctx_req.description(desc);
482    }
483    client.create_context(ctx_req).await?;
484
485    // 2. Fetch VTA config for URL/DID (needed to build the admin credential).
486    let config = client.get_config().await?;
487    let vta_did = config
488        .community_vta_did
489        .clone()
490        .ok_or("VTA DID not configured — cannot mint admin credential")?;
491    let vta_url = config.public_url.clone();
492
493    // 3. Generate admin did:key locally (private key never crosses the wire)
494    //    and register it with the VTA via POST /acl. This replaces the
495    //    pre-5c6 `POST /auth/credentials` round-trip that returned a base64
496    //    CredentialBundle in a plaintext JSON body.
497    eprintln!("Minting local admin credential and registering ACL...");
498    let (admin_credential, admin_did) =
499        crate::local_keygen::generate_admin_did_key(vta_did, vta_url);
500    let mut acl_req =
501        vta_sdk::client::CreateAclRequest::new(&admin_did, "admin").contexts(vec![id.to_string()]);
502    if let Some(l) = admin_label {
503        acl_req = acl_req.label(l);
504    }
505    client.create_acl(acl_req).await?;
506
507    // 4. Optionally create a DID and collect its secrets
508    let provisioned_did = if let Some(opts) = did_opts {
509        eprintln!("Creating WebVH DID...");
510        let req = CreateDidWebvhRequest {
511            context_id: id.to_string(),
512            server_id: opts.server_id,
513            url: opts.did_url,
514            path: None,
515            label: Some(id.to_string()),
516            portable: opts.portable,
517            add_mediator_service: opts.add_mediator_service,
518            additional_services: None,
519            pre_rotation_count: opts.pre_rotation_count,
520            did_document: None,
521            did_log: None,
522            set_primary: true,
523            signing_key_id: None,
524            ka_key_id: None,
525            template: None,
526            template_context: None,
527            template_vars: std::collections::HashMap::new(),
528        };
529        let did_result = client.create_did_webvh(req).await?;
530
531        // Collect secrets for the DID keys
532        eprintln!("Fetching DID key secrets...");
533        let mut secrets: Vec<SecretEntry> = Vec::new();
534        // Signing key
535        secrets.push(
536            client
537                .get_key_secret(&did_result.signing_key_id)
538                .await?
539                .into(),
540        );
541        // Key-agreement key
542        secrets.push(client.get_key_secret(&did_result.ka_key_id).await?.into());
543        // Pre-rotation keys
544        for i in 0..did_result.pre_rotation_key_count {
545            let pre_rot_id = format!("{}#pre-rotation-{i}", did_result.did);
546            secrets.push(client.get_key_secret(&pre_rot_id).await?.into());
547        }
548
549        Some(ProvisionedDid {
550            id: did_result.did,
551            did_document: did_result.did_document,
552            log_entry: did_result.log_entry,
553            secrets,
554        })
555    } else {
556        None
557    };
558
559    // 5. Build the provision bundle
560    let bundle = ContextProvisionBundle {
561        context_id: id.to_string(),
562        context_name: name.to_string(),
563        vta_url: config.public_url,
564        vta_did: config.community_vta_did,
565        credential: admin_credential,
566        admin_did,
567        did: provisioned_did,
568    };
569
570    // 6. Seal and emit via the shared helper
571    crate::sealed_producer::emit_context_provision_bundle(bundle, &recipient, None).await
572}
573
574/// Build a `CredentialBundle` from a VTA-stored key, deriving its `did:key`.
575///
576/// The REST fetch is transport-specific; the derivation is shared via
577/// [`CredentialBundle::from_ed25519_seed_multibase`] with the offline
578/// path in `vta-service::operations::export::credential_from_key_offline`
579/// so the `did:key` encoding and bundle shape can't drift.
580async fn credential_from_key(
581    client: &VtaClient,
582    key_id: &str,
583    vta_did: &str,
584    vta_url: Option<&str>,
585) -> Result<(CredentialBundle, String), Box<dyn std::error::Error>> {
586    let secret = client.get_key_secret(key_id).await?;
587    CredentialBundle::from_ed25519_seed_multibase(&secret.private_key_multibase, vta_did, vta_url)
588        .map_err(|e| format!("Cannot decode key secret: {e}").into())
589}
590
591pub async fn cmd_context_reprovision(
592    client: &VtaClient,
593    id: &str,
594    key_id: Option<String>,
595    admin_label: Option<String>,
596    recipient: SealedRecipient,
597) -> Result<(), Box<dyn std::error::Error>> {
598    // 1. Fetch the existing context
599    eprintln!("Fetching context '{id}'...");
600    let ctx = client.get_context(id).await?;
601
602    // 2. Fetch VTA config for URL/DID
603    let config = client.get_config().await?;
604    let vta_did = config
605        .community_vta_did
606        .as_deref()
607        .ok_or("VTA DID not configured")?;
608
609    // 3. Resolve admin credential
610    let (admin_credential, admin_did) = if let Some(ref kid) = key_id {
611        // Direct key ID specified
612        eprintln!("Using key '{kid}'...");
613        credential_from_key(client, kid, vta_did, config.public_url.as_deref()).await?
614    } else {
615        // Interactive: list existing Ed25519 keys and let user choose
616        let keys_resp = client.list_keys(0, 10000, Some("active"), Some(id)).await?;
617        let ed25519_keys: Vec<_> = keys_resp
618            .keys
619            .iter()
620            .filter(|k| k.key_type == KeyType::Ed25519)
621            .collect();
622
623        eprintln!();
624        eprintln!("Select an admin credential key for context '{id}':");
625        eprintln!();
626        for (i, key) in ed25519_keys.iter().enumerate() {
627            let label = key
628                .label
629                .as_deref()
630                .map(|l| format!(" ({l})"))
631                .unwrap_or_default();
632            eprintln!("  [{}] {}{}", i + 1, key.key_id, label);
633        }
634        let new_option = ed25519_keys.len() + 1;
635        eprintln!("  [{}] Create a new admin key", new_option);
636        eprintln!();
637        eprint!("Choice [{}]: ", new_option);
638        io::stderr().flush()?;
639
640        let mut input = String::new();
641        io::stdin().read_line(&mut input)?;
642        let input = input.trim();
643
644        // Default to creating a new key if empty
645        let choice: usize = if input.is_empty() {
646            new_option
647        } else {
648            input
649                .parse()
650                .map_err(|_| format!("Invalid choice: {input}"))?
651        };
652
653        if choice == new_option {
654            // Create a new Ed25519 key in VTA scoped to this context
655            eprintln!("Creating new admin key...");
656            let key_resp = client
657                .create_key(CreateKeyRequest {
658                    key_type: KeyType::Ed25519,
659                    derivation_path: None,
660                    key_id: None,
661                    mnemonic: None,
662                    label: admin_label.or_else(|| Some("admin".to_string())),
663                    context_id: Some(id.to_string()),
664                })
665                .await?;
666            credential_from_key(
667                client,
668                &key_resp.key_id,
669                vta_did,
670                config.public_url.as_deref(),
671            )
672            .await?
673        } else if choice >= 1 && choice <= ed25519_keys.len() {
674            let selected = &ed25519_keys[choice - 1];
675            eprintln!("Using key '{}'...", selected.key_id);
676            credential_from_key(
677                client,
678                &selected.key_id,
679                vta_did,
680                config.public_url.as_deref(),
681            )
682            .await?
683        } else {
684            return Err(format!("Invalid choice: {choice}").into());
685        }
686    };
687
688    // 4. Ensure an ACL entry exists for this admin DID
689    if client.get_acl(&admin_did).await.is_err() {
690        eprintln!("Creating ACL entry for {admin_did}...");
691        client
692            .create_acl(
693                vta_sdk::client::CreateAclRequest::new(&admin_did, "admin")
694                    .contexts(vec![id.to_string()]),
695            )
696            .await?;
697    }
698
699    // 5. Collect DID material (document, log, secrets) when the context has a DID
700    let provisioned_did = if let Some(ref did_id) = ctx.did {
701        eprintln!("Fetching DID material...");
702
703        // Fetch the DID log and extract the DID document from it
704        let log_resp = client.get_did_webvh_log(did_id).await?;
705        let (did_document, log_entry) = if let Some(ref log_str) = log_resp.log {
706            let parsed: serde_json::Value = serde_json::from_str(log_str)
707                .map_err(|e| format!("failed to parse DID log: {e}"))?;
708            let doc = parsed.get("state").cloned();
709            (doc, Some(log_str.clone()))
710        } else {
711            (None, None)
712        };
713
714        // Fetch all active key secrets for this context
715        let secrets_bundle = client.fetch_did_secrets_bundle(id).await?;
716
717        Some(ProvisionedDid {
718            id: did_id.clone(),
719            did_document,
720            log_entry,
721            secrets: secrets_bundle.secrets,
722        })
723    } else {
724        None
725    };
726
727    // 6. Build the provision bundle
728    let bundle = ContextProvisionBundle {
729        context_id: id.to_string(),
730        context_name: ctx.name.clone(),
731        vta_url: config.public_url,
732        vta_did: config.community_vta_did,
733        credential: admin_credential,
734        admin_did,
735        did: provisioned_did,
736    };
737
738    // 7. Seal and emit via the shared helper
739    crate::sealed_producer::emit_context_provision_bundle(bundle, &recipient, None).await
740}