1 use std::{path::PathBuf, time::Duration};
4 use jsonrpsee::core::client::{Error as RpcError, Subscription, SubscriptionClientT};
6 use tokio::{select, time::sleep};
10 use crate::cli::{CliCommands, GroupPermission, LinkState};
11 use crate::jsonrpc::RpcClient;
14 #[allow(non_snake_case, clippy::too_many_arguments)]
18 const DEFAULT_TCP: &str = "127.0.0.1:7583";
19 const DEFAULT_SOCKET_SUFFIX: &str = "signal-cli/socket";
20 const DEFAULT_HTTP: &str = "http://localhost:8080/api/v1/rpc";
23 async fn main() -> Result<(), anyhow::Error> {
24 let cli = cli::Cli::parse();
26 let result = connect(cli).await;
30 Ok(v) => println!("{v}"),
31 Err(e) => return Err(anyhow::anyhow!("JSON-RPC command failed: {e:?}")),
36 async fn handle_command(
38 client: impl SubscriptionClientT + Sync,
39 ) -> Result<Value, RpcError> {
41 CliCommands::Receive { timeout } => {
42 let mut stream = client.subscribe_receive(cli.account).await?;
45 while let Some(v) = stream_next(timeout, &mut stream).await {
50 stream.unsubscribe().await?;
53 CliCommands::AddDevice { uri } => client.add_device(cli.account, uri).await,
57 } => client.block(cli.account, recipient, group_id).await,
58 CliCommands::DeleteLocalAccountData { ignore_registered } => {
60 .delete_local_account_data(cli.account, ignore_registered)
63 CliCommands::GetUserStatus { recipient } => {
64 client.get_user_status(cli.account, recipient).await
66 CliCommands::JoinGroup { uri } => client.join_group(cli.account, uri).await,
67 CliCommands::Link { name } => {
69 .start_link(cli.account)
71 .map_err(|e| RpcError::Custom(format!("JSON-RPC command startLink failed: {e:?}")))?
74 client.finish_link(url, name).await
76 CliCommands::ListAccounts => client.list_accounts().await,
77 CliCommands::ListContacts {
84 .list_contacts(cli.account, recipient, all_recipients, blocked, name)
87 CliCommands::ListDevices => client.list_devices(cli.account).await,
88 CliCommands::ListGroups {
91 } => client.list_groups(cli.account, group_id).await,
92 CliCommands::ListIdentities { number } => client.list_identities(cli.account, number).await,
93 CliCommands::ListStickerPacks => client.list_sticker_packs(cli.account).await,
94 CliCommands::QuitGroup {
100 .quit_group(cli.account, group_id, delete, admin)
103 CliCommands::Register { voice, captcha } => {
104 client.register(cli.account, voice, captcha).await
106 CliCommands::RemoveContact {
112 .remove_contact(cli.account, recipient, forget, hide)
115 CliCommands::RemoveDevice { device_id } => {
116 client.remove_device(cli.account, device_id).await
118 CliCommands::RemovePin => client.remove_pin(cli.account).await,
119 CliCommands::RemoteDelete {
166 message.unwrap_or_default(),
187 CliCommands::SendContacts => client.send_contacts(cli.account).await,
188 CliCommands::SendPaymentNotification {
194 .send_payment_notification(cli.account, recipient, receipt, note)
197 CliCommands::SendReaction {
221 CliCommands::SendReceipt {
232 cli::ReceiptType::Read => "read".to_owned(),
233 cli::ReceiptType::Viewed => "viewed".to_owned(),
238 CliCommands::SendSyncRequest => client.send_sync_request(cli.account).await,
239 CliCommands::SendTyping {
245 .send_typing(cli.account, recipient, group_id, stop)
248 CliCommands::SetPin { pin } => client.set_pin(cli.account, pin).await,
249 CliCommands::SubmitRateLimitChallenge { challenge, captcha } => {
251 .submit_rate_limit_challenge(cli.account, challenge, captcha)
256 trust_all_known_keys,
257 verified_safety_number,
263 trust_all_known_keys,
264 verified_safety_number,
268 CliCommands::Unblock {
271 } => client.unblock(cli.account, recipient, group_id).await,
272 CliCommands::Unregister { delete_account } => {
273 client.unregister(cli.account, delete_account).await
275 CliCommands::UpdateAccount {
277 unrestricted_unidentified_sender,
278 discoverable_by_number,
285 unrestricted_unidentified_sender,
286 discoverable_by_number,
291 CliCommands::UpdateConfiguration {
293 unidentified_delivery_indicators,
298 .update_configuration(
301 unidentified_delivery_indicators,
307 CliCommands::UpdateContact {
313 .update_contact(cli.account, recipient, name, expiration)
316 CliCommands::UpdateGroup {
329 set_permission_add_member,
330 set_permission_edit_details,
331 set_permission_send_messages,
348 link.map(|link| match link {
349 LinkState::Enabled => "enabled".to_owned(),
350 LinkState::EnabledWithApproval => "enabledWithApproval".to_owned(),
351 LinkState::Disabled => "disabled".to_owned(),
353 set_permission_add_member.map(|p| match p {
354 GroupPermission::EveryMember => "everyMember".to_owned(),
355 GroupPermission::OnlyAdmins => "onlyAdmins".to_owned(),
357 set_permission_edit_details.map(|p| match p {
358 GroupPermission::EveryMember => "everyMember".to_owned(),
359 GroupPermission::OnlyAdmins => "onlyAdmins".to_owned(),
361 set_permission_send_messages.map(|p| match p {
362 GroupPermission::EveryMember => "everyMember".to_owned(),
363 GroupPermission::OnlyAdmins => "onlyAdmins".to_owned(),
369 CliCommands::UpdateProfile {
391 CliCommands::UploadStickerPack { path } => {
392 client.upload_sticker_pack(cli.account, path).await
394 CliCommands::Verify {
397 } => client.verify(cli.account, verification_code, pin).await,
398 CliCommands::Version => client.version().await,
399 CliCommands::AddStickerPack { uri } => client.add_sticker_pack(cli.account, uri).await,
400 CliCommands::FinishChangeNumber {
406 .finish_change_number(cli.account, number, verification_code, pin)
409 CliCommands::GetAttachment {
415 .get_attachment(cli.account, id, recipient, group_id)
418 CliCommands::GetAvatar {
424 .get_avatar(cli.account, contact, profile, group_id)
427 CliCommands::GetSticker {
430 } => client.get_sticker(cli.account, pack_id, sticker_id).await,
431 CliCommands::StartChangeNumber {
437 .start_change_number(cli.account, number, voice, captcha)
443 async fn connect(cli: Cli) -> Result<Value, RpcError> {
444 if let Some(http) = &cli.json_rpc_http {
445 let uri = if let Some(uri) = http {
450 let client = jsonrpc::connect_http(uri)
452 .map_err(|e| RpcError::Custom(format!("Failed to connect to socket: {e}")))?;
454 handle_command(cli, client).await
455 } else if let Some(tcp) = cli.json_rpc_tcp {
456 let socket_addr = tcp.unwrap_or_else(|| DEFAULT_TCP.parse().unwrap());
457 let client = jsonrpc::connect_tcp(socket_addr)
459 .map_err(|e| RpcError::Custom(format!("Failed to connect to socket: {e}")))?;
461 handle_command(cli, client).await
463 let socket_path = cli
468 std::env::var_os("XDG_RUNTIME_DIR").map(|runtime_dir| {
469 PathBuf::from(runtime_dir)
470 .join(DEFAULT_SOCKET_SUFFIX)
474 .unwrap_or_else(|| ("/run".to_owned() + DEFAULT_SOCKET_SUFFIX).into());
475 let client = jsonrpc::connect_unix(socket_path)
477 .map_err(|e| RpcError::Custom(format!("Failed to connect to socket: {e}")))?;
479 handle_command(cli, client).await
483 async fn stream_next(
485 stream: &mut Subscription<Value>,
486 ) -> Option<Result<Value, RpcError>> {
491 v = stream.next() => v,
492 _= sleep(Duration::from_millis((timeout * 1000.0) as u64)) => None,