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 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 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
78pub 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 let table = Table::new(
143 rows,
144 [
145 Constraint::Min(16), Constraint::Min(20), Constraint::Min(40), Constraint::Length(16), Constraint::Length(10), ],
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
164pub 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#[derive(Debug, Default, Clone)]
216pub struct AdminAclOptions {
217 pub did: Option<String>,
219 pub label: Option<String>,
221 pub expires_at: Option<u64>,
223 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 parent: Option<String>,
241 admin: AdminAclOptions,
242) -> Result<(), Box<dyn std::error::Error>> {
243 use crate::render::{RESET, YELLOW};
244 use vta_sdk::error::VtaError;
245
246 let effective_id = parent
249 .as_ref()
250 .map_or_else(|| id.to_string(), |p| format!("{p}/{id}"));
251 let req = CreateContextRequest {
252 id: id.to_string(),
253 name: name.to_string(),
254 description,
255 parent,
256 };
257 let resp = match client.create_context(req).await {
258 Ok(r) => r,
259 Err(VtaError::Conflict(_)) if admin.is_requested() => {
265 let did = admin.did.as_deref().unwrap_or_default();
266 let bin = crate::render::bin_name();
267 eprintln!(
268 "{YELLOW}\u{26a0}{RESET} Context '{effective_id}' already exists — skipping context creation."
269 );
270 eprintln!();
271 eprintln!(" The --admin-did was NOT added. To grant admin access to an existing");
272 eprintln!(" context, use the ACL command directly:");
273 eprintln!();
274 let mut hint =
275 format!(" {bin} acl create --did {did} --role admin --contexts {effective_id}");
276 if let Some(label) = admin.label.as_deref() {
277 hint.push_str(&format!(" --label '{label}'"));
278 }
279 match (admin.expires_duration.as_deref(), admin.expires_at) {
280 (Some(raw), _) => hint.push_str(&format!(" --expires {raw}")),
284 (None, Some(expires_at)) => {
285 let remaining = expires_at.saturating_sub(crate::duration::now_unix());
286 hint.push_str(&format!(" --expires {remaining}s"));
287 }
288 (None, None) => {}
289 }
290 eprintln!("{hint}");
291 return Ok(());
292 }
293 Err(e) => return Err(e.into()),
294 };
295 println!("Context created:");
296 println!(" ID: {}", resp.id);
297 println!(" Name: {}", resp.name);
298 println!(" Base Path: {}", resp.base_path);
299
300 if admin.is_requested() {
301 let did = admin.did.as_deref().unwrap_or_default();
302 if !did.starts_with("did:") {
303 return Err(format!(
304 "--admin-did must start with `did:` (got {did:?}) — context was created but no ACL entry was added"
305 )
306 .into());
307 }
308 let mut acl_req =
311 vta_sdk::client::CreateAclRequest::new(did, "admin").contexts(vec![resp.id.clone()]);
312 if let Some(label) = admin.label.as_deref() {
313 acl_req = acl_req.label(label);
314 }
315 if let Some(expires_at) = admin.expires_at {
316 acl_req = acl_req.expires_at(expires_at);
317 }
318 let acl = client.create_acl(acl_req).await?;
319
320 println!();
321 println!("Admin ACL entry created:");
322 println!(" DID: {}", acl.did);
323 println!(" Role: {}", acl.role);
324 println!(" Contexts: {}", acl.allowed_contexts.join(", "));
325 if let Some(ref label) = acl.label {
326 println!(" Label: {label}");
327 }
328 match acl.expires_at {
329 Some(secs) => {
330 println!(
331 " Expires at: {} ({}) — setup ACL",
332 crate::duration::format_local_time(secs),
333 crate::duration::format_remaining(secs),
334 );
335 println!();
336 println!(" The admin should authenticate before expiry. On first successful");
337 println!(" connect PNM rotates to a fresh long-lived did:key and replaces this");
338 println!(" temporary entry with a permanent one.");
339 }
340 None => println!(" Expires at: (permanent)"),
341 }
342 }
343
344 Ok(())
345}
346
347pub async fn cmd_context_update(
348 client: &VtaClient,
349 id: &str,
350 name: Option<String>,
351 did: Option<String>,
352 description: Option<String>,
353) -> Result<(), Box<dyn std::error::Error>> {
354 let req = UpdateContextRequest {
355 name,
356 did,
357 description,
358 };
359 let resp = client.update_context(id, req).await?;
360 println!("Context updated:");
361 render_context_record(&resp);
362 Ok(())
363}
364
365pub async fn cmd_context_update_did(
366 client: &VtaClient,
367 id: &str,
368 did: &str,
369) -> Result<(), Box<dyn std::error::Error>> {
370 let resp = client.update_context_did(id, did).await?;
371 println!("Context DID updated:");
372 println!(" ID: {}", resp.id);
373 println!(
374 " DID: {}",
375 resp.did.as_deref().unwrap_or("(not set)")
376 );
377 println!(
378 " Updated At: {}",
379 crate::duration::format_local_datetime(resp.updated_at)
380 );
381 Ok(())
382}
383
384pub fn render_delete_context_preview(
391 id: &str,
392 preview: &vta_sdk::protocols::context_management::delete::DeleteContextPreviewResultBody,
393) -> bool {
394 let has_resources = !preview.keys.is_empty()
395 || !preview.webvh_dids.is_empty()
396 || !preview.acl_entries_removed.is_empty()
397 || !preview.acl_entries_updated.is_empty();
398
399 if !has_resources {
400 return false;
401 }
402
403 println!(
404 "Deleting context '{}' will remove the following resources:\n",
405 id
406 );
407
408 if !preview.keys.is_empty() {
409 println!(" Keys ({}):", preview.keys.len());
410 for key in &preview.keys {
411 println!(" - {key}");
412 }
413 }
414
415 if !preview.webvh_dids.is_empty() {
416 println!(" WebVH DIDs ({}):", preview.webvh_dids.len());
417 for did in &preview.webvh_dids {
418 println!(" - {did}");
419 }
420 }
421
422 if !preview.acl_entries_removed.is_empty() {
423 println!(
424 " ACL entries removed ({}):",
425 preview.acl_entries_removed.len()
426 );
427 for did in &preview.acl_entries_removed {
428 println!(" - {did}");
429 }
430 }
431
432 if !preview.acl_entries_updated.is_empty() {
433 println!(
434 " ACL entries updated (context removed from access list) ({}):",
435 preview.acl_entries_updated.len()
436 );
437 for did in &preview.acl_entries_updated {
438 println!(" - {did}");
439 }
440 }
441
442 println!();
443 true
444}
445
446pub fn confirm_destructive(prompt: &str) -> Result<bool, Box<dyn std::error::Error>> {
450 print!("{prompt} [y/N] ");
451 io::stdout().flush()?;
452 let mut input = String::new();
453 io::stdin().read_line(&mut input)?;
454 let input = input.trim().to_lowercase();
455 Ok(input == "y" || input == "yes")
456}
457
458pub async fn cmd_context_delete(
459 client: &VtaClient,
460 id: &str,
461 force: bool,
462) -> Result<(), Box<dyn std::error::Error>> {
463 let preview = client.preview_delete_context(id).await?;
465
466 let has_resources = render_delete_context_preview(id, &preview);
467
468 if has_resources && !force && !confirm_destructive("Proceed with deletion?")? {
469 println!("Aborted.");
470 return Ok(());
471 }
472
473 client.delete_context(id, true).await?;
474 println!("Context deleted: {id}");
475 Ok(())
476}
477
478pub async fn cmd_context_provision(
479 client: &VtaClient,
480 id: &str,
481 name: &str,
482 description: Option<String>,
483 admin_label: Option<String>,
484 did_opts: Option<ProvisionDidOptions>,
485 recipient: SealedRecipient,
486) -> Result<(), Box<dyn std::error::Error>> {
487 eprintln!("Creating context '{id}'...");
489 let mut ctx_req = CreateContextRequest::new(id, name);
490 if let Some(desc) = description {
491 ctx_req = ctx_req.description(desc);
492 }
493 client.create_context(ctx_req).await?;
494
495 let config = client.get_config().await?;
497 let vta_did = config
498 .community_vta_did
499 .clone()
500 .ok_or("VTA DID not configured — cannot mint admin credential")?;
501 let vta_url = config.public_url.clone();
502
503 eprintln!("Minting local admin credential and registering ACL...");
508 let (admin_credential, admin_did) =
509 crate::local_keygen::generate_admin_did_key(vta_did, vta_url);
510 let mut acl_req =
511 vta_sdk::client::CreateAclRequest::new(&admin_did, "admin").contexts(vec![id.to_string()]);
512 if let Some(l) = admin_label {
513 acl_req = acl_req.label(l);
514 }
515 client.create_acl(acl_req).await?;
516
517 let provisioned_did = if let Some(opts) = did_opts {
519 eprintln!("Creating WebVH DID...");
520 let req = CreateDidWebvhRequest {
521 context_id: id.to_string(),
522 server_id: opts.server_id,
523 url: opts.did_url,
524 path: None,
525 path_mode: None,
528 domain: None,
531 label: Some(id.to_string()),
532 portable: opts.portable,
533 add_mediator_service: opts.add_mediator_service,
534 additional_services: None,
535 pre_rotation_count: opts.pre_rotation_count,
536 did_document: None,
537 did_log: None,
538 set_primary: true,
539 signing_key_id: None,
540 ka_key_id: None,
541 template: None,
542 template_context: None,
543 template_vars: std::collections::HashMap::new(),
544 };
545 let did_result = client.create_did_webvh(req).await?;
546
547 eprintln!("Fetching DID key secrets...");
549 let mut secrets: Vec<SecretEntry> = Vec::new();
550 secrets.push(
552 client
553 .get_key_secret(&did_result.signing_key_id)
554 .await?
555 .into(),
556 );
557 secrets.push(client.get_key_secret(&did_result.ka_key_id).await?.into());
559 for i in 0..did_result.pre_rotation_key_count {
561 let pre_rot_id = format!("{}#pre-rotation-{i}", did_result.did);
562 secrets.push(client.get_key_secret(&pre_rot_id).await?.into());
563 }
564
565 Some(ProvisionedDid {
566 id: did_result.did,
567 did_document: did_result.did_document,
568 log_entry: did_result.log_entry,
569 secrets,
570 })
571 } else {
572 None
573 };
574
575 let bundle = ContextProvisionBundle {
577 context_id: id.to_string(),
578 context_name: name.to_string(),
579 vta_url: config.public_url,
580 vta_did: config.community_vta_did,
581 credential: admin_credential,
582 admin_did,
583 did: provisioned_did,
584 };
585
586 crate::sealed_producer::emit_context_provision_bundle(bundle, &recipient, None).await
588}
589
590async fn credential_from_key(
597 client: &VtaClient,
598 key_id: &str,
599 vta_did: &str,
600 vta_url: Option<&str>,
601) -> Result<(CredentialBundle, String), Box<dyn std::error::Error>> {
602 let secret = client.get_key_secret(key_id).await?;
603 CredentialBundle::from_ed25519_seed_multibase(&secret.private_key_multibase, vta_did, vta_url)
604 .map_err(|e| format!("Cannot decode key secret: {e}").into())
605}
606
607pub async fn cmd_context_reprovision(
608 client: &VtaClient,
609 id: &str,
610 key_id: Option<String>,
611 admin_label: Option<String>,
612 recipient: SealedRecipient,
613) -> Result<(), Box<dyn std::error::Error>> {
614 eprintln!("Fetching context '{id}'...");
616 let ctx = client.get_context(id).await?;
617
618 let config = client.get_config().await?;
620 let vta_did = config
621 .community_vta_did
622 .as_deref()
623 .ok_or("VTA DID not configured")?;
624
625 let (admin_credential, admin_did) = if let Some(ref kid) = key_id {
627 eprintln!("Using key '{kid}'...");
629 credential_from_key(client, kid, vta_did, config.public_url.as_deref()).await?
630 } else {
631 let keys_resp = client.list_keys(0, 10000, Some("active"), Some(id)).await?;
633 let ed25519_keys: Vec<_> = keys_resp
634 .keys
635 .iter()
636 .filter(|k| k.key_type == KeyType::Ed25519)
637 .collect();
638
639 eprintln!();
640 eprintln!("Select an admin credential key for context '{id}':");
641 eprintln!();
642 for (i, key) in ed25519_keys.iter().enumerate() {
643 let label = key
644 .label
645 .as_deref()
646 .map(|l| format!(" ({l})"))
647 .unwrap_or_default();
648 eprintln!(" [{}] {}{}", i + 1, key.key_id, label);
649 }
650 let new_option = ed25519_keys.len() + 1;
651 eprintln!(" [{}] Create a new admin key", new_option);
652 eprintln!();
653 eprint!("Choice [{}]: ", new_option);
654 io::stderr().flush()?;
655
656 let mut input = String::new();
657 io::stdin().read_line(&mut input)?;
658 let input = input.trim();
659
660 let choice: usize = if input.is_empty() {
662 new_option
663 } else {
664 input
665 .parse()
666 .map_err(|_| format!("Invalid choice: {input}"))?
667 };
668
669 if choice == new_option {
670 eprintln!("Creating new admin key...");
672 let key_resp = client
673 .create_key(CreateKeyRequest {
674 key_type: KeyType::Ed25519,
675 derivation_path: None,
676 key_id: None,
677 mnemonic: None,
678 label: admin_label.or_else(|| Some("admin".to_string())),
679 context_id: Some(id.to_string()),
680 })
681 .await?;
682 credential_from_key(
683 client,
684 &key_resp.key_id,
685 vta_did,
686 config.public_url.as_deref(),
687 )
688 .await?
689 } else if choice >= 1 && choice <= ed25519_keys.len() {
690 let selected = &ed25519_keys[choice - 1];
691 eprintln!("Using key '{}'...", selected.key_id);
692 credential_from_key(
693 client,
694 &selected.key_id,
695 vta_did,
696 config.public_url.as_deref(),
697 )
698 .await?
699 } else {
700 return Err(format!("Invalid choice: {choice}").into());
701 }
702 };
703
704 if client.get_acl(&admin_did).await.is_err() {
706 eprintln!("Creating ACL entry for {admin_did}...");
707 client
708 .create_acl(
709 vta_sdk::client::CreateAclRequest::new(&admin_did, "admin")
710 .contexts(vec![id.to_string()]),
711 )
712 .await?;
713 }
714
715 let provisioned_did = if let Some(ref did_id) = ctx.did {
717 eprintln!("Fetching DID material...");
718
719 let log_resp = client.get_did_webvh_log(did_id).await?;
721 let (did_document, log_entry) = if let Some(ref log_str) = log_resp.log {
722 let parsed: serde_json::Value = serde_json::from_str(log_str)
723 .map_err(|e| format!("failed to parse DID log: {e}"))?;
724 let doc = parsed.get("state").cloned();
725 (doc, Some(log_str.clone()))
726 } else {
727 (None, None)
728 };
729
730 let secrets_bundle = client.fetch_did_secrets_bundle(id).await?;
732
733 Some(ProvisionedDid {
734 id: did_id.clone(),
735 did_document,
736 log_entry,
737 secrets: secrets_bundle.secrets,
738 })
739 } else {
740 None
741 };
742
743 let bundle = ContextProvisionBundle {
745 context_id: id.to_string(),
746 context_name: ctx.name.clone(),
747 vta_url: config.public_url,
748 vta_did: config.community_vta_did,
749 credential: admin_credential,
750 admin_did,
751 did: provisioned_did,
752 };
753
754 crate::sealed_producer::emit_context_provision_bundle(bundle, &recipient, None).await
756}