528 lines
14 KiB
Rust
528 lines
14 KiB
Rust
use super::get_latest_event;
|
|
use crate::Nostr;
|
|
use keyring::Entry;
|
|
use nostr_sdk::prelude::*;
|
|
use std::{str::FromStr, time::Duration};
|
|
use tauri::State;
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn get_current_user_profile(state: State<'_, Nostr>) -> Result<String, String> {
|
|
let client = &state.client;
|
|
let signer = client.signer().await.unwrap();
|
|
let public_key = signer.public_key().await.unwrap();
|
|
let filter = Filter::new()
|
|
.author(public_key)
|
|
.kind(Kind::Metadata)
|
|
.limit(1);
|
|
|
|
match client
|
|
.get_events_of(vec![filter], Some(Duration::from_secs(10)))
|
|
.await
|
|
{
|
|
Ok(events) => {
|
|
if let Some(event) = get_latest_event(&events) {
|
|
if let Ok(metadata) = Metadata::from_json(&event.content) {
|
|
Ok(metadata.as_json())
|
|
} else {
|
|
Err("Parse metadata failed".into())
|
|
}
|
|
} else {
|
|
Err("Not found".into())
|
|
}
|
|
}
|
|
Err(_) => Err("Not found".into()),
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn get_profile(id: &str, state: State<'_, Nostr>) -> Result<String, String> {
|
|
let client = &state.client;
|
|
let public_key: Option<PublicKey> = match Nip19::from_bech32(id) {
|
|
Ok(val) => match val {
|
|
Nip19::Pubkey(pubkey) => Some(pubkey),
|
|
Nip19::Profile(profile) => {
|
|
let relays = profile.relays;
|
|
for relay in relays.into_iter() {
|
|
let _ = client.add_relay(&relay).await.unwrap_or_default();
|
|
client.connect_relay(&relay).await.unwrap_or_default();
|
|
}
|
|
Some(profile.public_key)
|
|
}
|
|
_ => None,
|
|
},
|
|
Err(_) => match PublicKey::from_str(id) {
|
|
Ok(val) => Some(val),
|
|
Err(_) => None,
|
|
},
|
|
};
|
|
|
|
if let Some(author) = public_key {
|
|
let filter = Filter::new().author(author).kind(Kind::Metadata).limit(1);
|
|
let query = client
|
|
.get_events_of(vec![filter], Some(Duration::from_secs(10)))
|
|
.await;
|
|
|
|
if let Ok(events) = query {
|
|
if let Some(event) = events.first() {
|
|
if let Ok(metadata) = Metadata::from_json(&event.content) {
|
|
Ok(metadata.as_json())
|
|
} else {
|
|
Err("Parse metadata failed".into())
|
|
}
|
|
} else {
|
|
let rand_metadata = Metadata::new();
|
|
Ok(rand_metadata.as_json())
|
|
}
|
|
} else {
|
|
Err("Get metadata failed".into())
|
|
}
|
|
} else {
|
|
Err("Public Key is not valid".into())
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn set_contact_list(pubkeys: Vec<&str>, state: State<'_, Nostr>) -> Result<bool, String> {
|
|
let client = &state.client;
|
|
let contact_list: Vec<Contact> = pubkeys
|
|
.into_iter()
|
|
.map(|p| Contact::new(PublicKey::from_hex(p).unwrap(), None, Some("")))
|
|
.collect();
|
|
|
|
match client.set_contact_list(contact_list).await {
|
|
Ok(_) => Ok(true),
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn get_contact_list(state: State<'_, Nostr>) -> Result<Vec<String>, String> {
|
|
let client = &state.client;
|
|
|
|
match client.get_contact_list(Some(Duration::from_secs(10))).await {
|
|
Ok(contact_list) => {
|
|
if !contact_list.is_empty() {
|
|
let list = contact_list
|
|
.into_iter()
|
|
.map(|f| f.public_key.to_hex())
|
|
.collect();
|
|
|
|
Ok(list)
|
|
} else {
|
|
Err("Empty.".into())
|
|
}
|
|
}
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn create_profile(
|
|
name: &str,
|
|
display_name: &str,
|
|
about: &str,
|
|
picture: &str,
|
|
banner: &str,
|
|
nip05: &str,
|
|
lud16: &str,
|
|
website: &str,
|
|
state: State<'_, Nostr>,
|
|
) -> Result<String, String> {
|
|
let client = &state.client;
|
|
let mut metadata = Metadata::new()
|
|
.name(name)
|
|
.display_name(display_name)
|
|
.about(about)
|
|
.nip05(nip05)
|
|
.lud16(lud16);
|
|
|
|
if let Ok(url) = Url::parse(picture) {
|
|
metadata = metadata.picture(url)
|
|
}
|
|
|
|
if let Ok(url) = Url::parse(banner) {
|
|
metadata = metadata.banner(url)
|
|
}
|
|
|
|
if let Ok(url) = Url::parse(website) {
|
|
metadata = metadata.website(url)
|
|
}
|
|
|
|
if let Ok(event_id) = client.set_metadata(&metadata).await {
|
|
Ok(event_id.to_string())
|
|
} else {
|
|
Err("Create profile failed".into())
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn follow(
|
|
id: &str,
|
|
alias: Option<&str>,
|
|
state: State<'_, Nostr>,
|
|
) -> Result<String, String> {
|
|
let client = &state.client;
|
|
let public_key = PublicKey::from_str(id).unwrap();
|
|
let contact = Contact::new(public_key, None, alias); // #TODO: Add relay_url
|
|
let contact_list = client.get_contact_list(Some(Duration::from_secs(10))).await;
|
|
|
|
match contact_list {
|
|
Ok(mut old_list) => {
|
|
old_list.push(contact);
|
|
let new_list = old_list.into_iter();
|
|
|
|
match client.set_contact_list(new_list).await {
|
|
Ok(event_id) => Ok(event_id.to_string()),
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn unfollow(id: &str, state: State<'_, Nostr>) -> Result<String, String> {
|
|
let client = &state.client;
|
|
let public_key = PublicKey::from_str(id).unwrap();
|
|
let contact_list = client.get_contact_list(Some(Duration::from_secs(10))).await;
|
|
|
|
match contact_list {
|
|
Ok(mut old_list) => {
|
|
let index = old_list
|
|
.iter()
|
|
.position(|x| x.public_key == public_key)
|
|
.unwrap();
|
|
old_list.remove(index);
|
|
|
|
let new_list = old_list.into_iter();
|
|
|
|
match client.set_contact_list(new_list).await {
|
|
Ok(event_id) => Ok(event_id.to_string()),
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn set_nstore(
|
|
key: &str,
|
|
content: &str,
|
|
state: State<'_, Nostr>,
|
|
) -> Result<String, String> {
|
|
let client = &state.client;
|
|
|
|
match client.signer().await {
|
|
Ok(signer) => {
|
|
let public_key = signer.public_key().await.unwrap();
|
|
let encrypted = signer.nip44_encrypt(public_key, content).await.unwrap();
|
|
|
|
let tag = Tag::identifier(key);
|
|
let builder = EventBuilder::new(Kind::ApplicationSpecificData, encrypted, vec![tag]);
|
|
|
|
match client.send_event_builder(builder).await {
|
|
Ok(event_id) => Ok(event_id.to_string()),
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn get_nstore(key: &str, state: State<'_, Nostr>) -> Result<String, String> {
|
|
let client = &state.client;
|
|
|
|
if let Ok(signer) = client.signer().await {
|
|
let public_key = signer.public_key().await.unwrap();
|
|
let filter = Filter::new()
|
|
.author(public_key)
|
|
.kind(Kind::ApplicationSpecificData)
|
|
.identifier(key)
|
|
.limit(1);
|
|
|
|
match client
|
|
.get_events_of(vec![filter], Some(Duration::from_secs(5)))
|
|
.await
|
|
{
|
|
Ok(events) => {
|
|
if let Some(event) = events.first() {
|
|
let content = event.content();
|
|
match signer.nip44_decrypt(public_key, content).await {
|
|
Ok(decrypted) => Ok(decrypted),
|
|
Err(_) => Err(event.content.to_string()),
|
|
}
|
|
} else {
|
|
Err("Value not found".into())
|
|
}
|
|
}
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
} else {
|
|
Err("Signer is required".into())
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn set_nwc(uri: &str, state: State<'_, Nostr>) -> Result<bool, String> {
|
|
let client = &state.client;
|
|
|
|
if let Ok(nwc_uri) = NostrWalletConnectURI::from_str(uri) {
|
|
let nwc = NWC::new(nwc_uri);
|
|
let keyring = Entry::new("Lume Secret Storage", "NWC").unwrap();
|
|
let _ = keyring.set_password(uri);
|
|
let _ = client.set_zapper(nwc).await;
|
|
|
|
Ok(true)
|
|
} else {
|
|
Err("Set NWC failed".into())
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn load_nwc(state: State<'_, Nostr>) -> Result<bool, String> {
|
|
let client = &state.client;
|
|
let keyring = Entry::new("Lume Secret Storage", "NWC").unwrap();
|
|
|
|
match keyring.get_password() {
|
|
Ok(val) => {
|
|
let uri = NostrWalletConnectURI::from_str(&val).unwrap();
|
|
let nwc = NWC::new(uri);
|
|
client.set_zapper(nwc).await;
|
|
|
|
Ok(true)
|
|
}
|
|
Err(_) => Ok(false),
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn get_balance() -> Result<String, String> {
|
|
let keyring = Entry::new("Lume Secret Storage", "NWC").unwrap();
|
|
|
|
match keyring.get_password() {
|
|
Ok(val) => {
|
|
let uri = NostrWalletConnectURI::from_str(&val).unwrap();
|
|
let nwc = NWC::new(uri);
|
|
if let Ok(balance) = nwc.get_balance().await {
|
|
Ok(balance.to_string())
|
|
} else {
|
|
Err("Get balance failed".into())
|
|
}
|
|
}
|
|
Err(_) => Err("Something wrong".into()),
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn zap_profile(
|
|
id: &str,
|
|
amount: &str,
|
|
message: &str,
|
|
state: State<'_, Nostr>,
|
|
) -> Result<bool, String> {
|
|
let client = &state.client;
|
|
let public_key: Option<PublicKey> = match Nip19::from_bech32(id) {
|
|
Ok(val) => match val {
|
|
Nip19::Pubkey(pubkey) => Some(pubkey),
|
|
Nip19::Profile(profile) => Some(profile.public_key),
|
|
_ => None,
|
|
},
|
|
Err(_) => match PublicKey::from_str(id) {
|
|
Ok(val) => Some(val),
|
|
Err(_) => None,
|
|
},
|
|
};
|
|
|
|
if let Some(recipient) = public_key {
|
|
let details = ZapDetails::new(ZapType::Public).message(message);
|
|
let num = amount.parse::<u64>().unwrap();
|
|
|
|
if (client.zap(recipient, num, Some(details)).await).is_ok() {
|
|
Ok(true)
|
|
} else {
|
|
Err("Zap profile failed".into())
|
|
}
|
|
} else {
|
|
Err("Parse public key failed".into())
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn zap_event(
|
|
id: &str,
|
|
amount: &str,
|
|
message: &str,
|
|
state: State<'_, Nostr>,
|
|
) -> Result<bool, String> {
|
|
let client = &state.client;
|
|
let event_id: Option<EventId> = match Nip19::from_bech32(id) {
|
|
Ok(val) => match val {
|
|
Nip19::EventId(id) => Some(id),
|
|
Nip19::Event(event) => Some(event.event_id),
|
|
_ => None,
|
|
},
|
|
Err(_) => match EventId::from_hex(id) {
|
|
Ok(val) => Some(val),
|
|
Err(_) => None,
|
|
},
|
|
};
|
|
|
|
if let Some(recipient) = event_id {
|
|
let details = ZapDetails::new(ZapType::Public).message(message);
|
|
let num = amount.parse::<u64>().unwrap();
|
|
|
|
if (client.zap(recipient, num, Some(details)).await).is_ok() {
|
|
Ok(true)
|
|
} else {
|
|
Err("Zap event failed".into())
|
|
}
|
|
} else {
|
|
Err("Parse public key failed".into())
|
|
}
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn friend_to_friend(npub: &str, state: State<'_, Nostr>) -> Result<bool, String> {
|
|
let client = &state.client;
|
|
|
|
match PublicKey::from_bech32(npub) {
|
|
Ok(author) => {
|
|
let mut contact_list: Vec<Contact> = Vec::new();
|
|
let contact_list_filter = Filter::new()
|
|
.author(author)
|
|
.kind(Kind::ContactList)
|
|
.limit(1);
|
|
|
|
if let Ok(contact_list_events) = client.get_events_of(vec![contact_list_filter], None).await {
|
|
for event in contact_list_events.into_iter() {
|
|
for tag in event.into_iter_tags() {
|
|
if let Some(TagStandard::PublicKey {
|
|
public_key,
|
|
relay_url,
|
|
alias,
|
|
uppercase: false,
|
|
}) = tag.to_standardized()
|
|
{
|
|
contact_list.push(Contact::new(public_key, relay_url, alias))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
match client.set_contact_list(contact_list).await {
|
|
Ok(_) => Ok(true),
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
|
|
pub async fn get_following(
|
|
state: State<'_, Nostr>,
|
|
public_key: &str,
|
|
timeout: Option<u64>,
|
|
) -> Result<Vec<String>, String> {
|
|
let client = &state.client;
|
|
let public_key = match PublicKey::from_str(public_key) {
|
|
Ok(val) => val,
|
|
Err(err) => return Err(err.to_string()),
|
|
};
|
|
let duration = timeout.map(Duration::from_secs);
|
|
let filter = Filter::new().kind(Kind::ContactList).author(public_key);
|
|
let events = match client.get_events_of(vec![filter], duration).await {
|
|
Ok(events) => events,
|
|
Err(err) => return Err(err.to_string()),
|
|
};
|
|
let mut ret: Vec<String> = vec![];
|
|
if let Some(latest_event) = events.iter().max_by_key(|event| event.created_at()) {
|
|
ret.extend(latest_event.tags().iter().filter_map(|tag| {
|
|
if let Some(TagStandard::PublicKey {
|
|
uppercase: false, ..
|
|
}) = <nostr_sdk::Tag as Clone>::clone(tag).to_standardized()
|
|
{
|
|
tag.content().map(String::from)
|
|
} else {
|
|
None
|
|
}
|
|
}));
|
|
}
|
|
Ok(ret)
|
|
}
|
|
|
|
pub async fn get_followers(
|
|
state: State<'_, Nostr>,
|
|
public_key: &str,
|
|
timeout: Option<u64>,
|
|
) -> Result<Vec<String>, String> {
|
|
let client = &state.client;
|
|
let public_key = match PublicKey::from_str(public_key) {
|
|
Ok(val) => val,
|
|
Err(err) => return Err(err.to_string()),
|
|
};
|
|
let duration = timeout.map(Duration::from_secs);
|
|
|
|
let filter = Filter::new().kind(Kind::ContactList).custom_tag(
|
|
SingleLetterTag::lowercase(Alphabet::P),
|
|
vec![public_key.to_hex()],
|
|
);
|
|
let events = match client.get_events_of(vec![filter], duration).await {
|
|
Ok(events) => events,
|
|
Err(err) => return Err(err.to_string()),
|
|
};
|
|
let ret: Vec<String> = events
|
|
.into_iter()
|
|
.map(|event| event.author().to_hex())
|
|
.collect();
|
|
Ok(ret)
|
|
//todo: get more than 500 events
|
|
}
|
|
|
|
#[tauri::command]
|
|
#[specta::specta]
|
|
pub async fn get_notifications(state: State<'_, Nostr>) -> Result<Vec<String>, String> {
|
|
let client = &state.client;
|
|
|
|
match client.signer().await {
|
|
Ok(signer) => {
|
|
let public_key = signer.public_key().await.unwrap();
|
|
let filter = Filter::new()
|
|
.pubkey(public_key)
|
|
.kinds(vec![
|
|
Kind::TextNote,
|
|
Kind::Repost,
|
|
Kind::Reaction,
|
|
Kind::ZapReceipt,
|
|
])
|
|
.limit(200);
|
|
|
|
match client
|
|
.database()
|
|
.query(vec![filter], Order::default())
|
|
.await
|
|
{
|
|
Ok(events) => Ok(events.into_iter().map(|ev| ev.as_json()).collect()),
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|
|
Err(err) => Err(err.to_string()),
|
|
}
|
|
}
|