mirror of
https://github.com/mimblewimble/grin-wallet.git
synced 2025-01-20 19:11:09 +03:00
Remove method from grin-wallet.yml and keybase.rs from adapters (#658)
* Remove method from grin-wallet.yml and keybase.rs from adapters * Remove keybase_notify_ttl from config * Remove keybase_notify_ttl from config * Fix tests Co-authored-by: rstetsurin <rstetsurin@coinloan.io>
This commit is contained in:
parent
a3687c69a8
commit
67f0e2b0e0
7 changed files with 3 additions and 505 deletions
|
@ -1164,8 +1164,7 @@ pub trait OwnerRpc {
|
||||||
"no_commit_cache": null,
|
"no_commit_cache": null,
|
||||||
"tls_certificate_file": null,
|
"tls_certificate_file": null,
|
||||||
"tls_certificate_key": null,
|
"tls_certificate_key": null,
|
||||||
"dark_background_color_scheme": null,
|
"dark_background_color_scheme": null
|
||||||
"keybase_notify_ttl": null
|
|
||||||
},
|
},
|
||||||
"logging_config": {
|
"logging_config": {
|
||||||
"log_to_stdout": false,
|
"log_to_stdout": false,
|
||||||
|
|
|
@ -109,16 +109,6 @@ fn comments() -> HashMap<String, String> {
|
||||||
"dark_background_color_scheme".to_string(),
|
"dark_background_color_scheme".to_string(),
|
||||||
"
|
"
|
||||||
#Whether to use the black background color scheme for command line
|
#Whether to use the black background color scheme for command line
|
||||||
"
|
|
||||||
.to_string(),
|
|
||||||
);
|
|
||||||
retval.insert(
|
|
||||||
"keybase_notify_ttl".to_string(),
|
|
||||||
"
|
|
||||||
#The exploding lifetime for keybase notification on coins received.
|
|
||||||
#Unit: Minute. Default value 1440 minutes for one day.
|
|
||||||
#Refer to https://keybase.io/blog/keybase-exploding-messages for detail.
|
|
||||||
#To disable this notification, set it as 0.
|
|
||||||
"
|
"
|
||||||
.to_string(),
|
.to_string(),
|
||||||
);
|
);
|
||||||
|
|
|
@ -51,8 +51,6 @@ pub struct WalletConfig {
|
||||||
/// Whether to use the black background color scheme for command line
|
/// Whether to use the black background color scheme for command line
|
||||||
/// if enabled, wallet command output color will be suitable for black background terminal
|
/// if enabled, wallet command output color will be suitable for black background terminal
|
||||||
pub dark_background_color_scheme: Option<bool>,
|
pub dark_background_color_scheme: Option<bool>,
|
||||||
/// The exploding lifetime (minutes) for keybase notification on coins received
|
|
||||||
pub keybase_notify_ttl: Option<u16>,
|
|
||||||
/// Scaling factor from transaction weight to transaction fee
|
/// Scaling factor from transaction weight to transaction fee
|
||||||
/// should match accept_fee_base parameter in grin-server
|
/// should match accept_fee_base parameter in grin-server
|
||||||
pub accept_fee_base: Option<u64>,
|
pub accept_fee_base: Option<u64>,
|
||||||
|
@ -73,7 +71,6 @@ impl Default for WalletConfig {
|
||||||
tls_certificate_file: None,
|
tls_certificate_file: None,
|
||||||
tls_certificate_key: None,
|
tls_certificate_key: None,
|
||||||
dark_background_color_scheme: Some(true),
|
dark_background_color_scheme: Some(true),
|
||||||
keybase_notify_ttl: Some(1440),
|
|
||||||
accept_fee_base: None,
|
accept_fee_base: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,477 +0,0 @@
|
||||||
// Copyright 2021 The Grin Developers
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
// Keybase Wallet Plugin
|
|
||||||
|
|
||||||
use crate::adapters::{SlateReceiver, SlateSender};
|
|
||||||
use crate::config::WalletConfig;
|
|
||||||
use crate::keychain::ExtKeychain;
|
|
||||||
use crate::libwallet::api_impl::foreign;
|
|
||||||
use crate::libwallet::{Error, ErrorKind, Slate, WalletInst};
|
|
||||||
use crate::util::ZeroingString;
|
|
||||||
use crate::{DefaultLCProvider, DefaultWalletImpl, HTTPNodeClient};
|
|
||||||
use serde::Serialize;
|
|
||||||
use serde_json::{from_str, json, to_string, Value};
|
|
||||||
use std::collections::{HashMap, HashSet};
|
|
||||||
use std::process::{Command, Stdio};
|
|
||||||
use std::str::from_utf8;
|
|
||||||
use std::thread::sleep;
|
|
||||||
use std::time::{Duration, Instant};
|
|
||||||
|
|
||||||
const TTL: u16 = 60; // TODO: Pass this as a parameter
|
|
||||||
const LISTEN_SLEEP_DURATION: Duration = Duration::from_millis(5000);
|
|
||||||
const POLL_SLEEP_DURATION: Duration = Duration::from_millis(1000);
|
|
||||||
|
|
||||||
// Which topic names to use for communication
|
|
||||||
const SLATE_NEW: &str = "grin_slate_new";
|
|
||||||
const SLATE_SIGNED: &str = "grin_slate_signed";
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct KeybaseChannel(String);
|
|
||||||
|
|
||||||
impl KeybaseChannel {
|
|
||||||
/// Check if keybase is installed and return an adapter object.
|
|
||||||
pub fn new(channel: String) -> Result<KeybaseChannel, Error> {
|
|
||||||
// Limit only one recipient
|
|
||||||
if channel.matches(',').count() > 0 {
|
|
||||||
return Err(
|
|
||||||
ErrorKind::GenericError("Only one recipient is supported!".to_owned()).into(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if !keybase_installed() {
|
|
||||||
return Err(ErrorKind::GenericError(
|
|
||||||
"Keybase executable not found, make sure it is installed and in your PATH"
|
|
||||||
.to_owned(),
|
|
||||||
)
|
|
||||||
.into());
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(KeybaseChannel(channel))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Check if keybase executable exists in path
|
|
||||||
fn keybase_installed() -> bool {
|
|
||||||
let mut proc = if cfg!(target_os = "windows") {
|
|
||||||
Command::new("where")
|
|
||||||
} else {
|
|
||||||
Command::new("which")
|
|
||||||
};
|
|
||||||
proc.arg("keybase").stdout(Stdio::null()).status().is_ok()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Send a json object to the keybase process. Type `keybase chat api --help` for a list of available methods.
|
|
||||||
fn api_send(payload: &str) -> Result<Value, Error> {
|
|
||||||
let mut proc = Command::new("keybase");
|
|
||||||
proc.args(&["chat", "api", "-m", &payload]);
|
|
||||||
let output = proc.output().expect("No output");
|
|
||||||
if !output.status.success() {
|
|
||||||
error!(
|
|
||||||
"keybase api fail: {} {}",
|
|
||||||
String::from_utf8_lossy(&output.stdout),
|
|
||||||
String::from_utf8_lossy(&output.stderr)
|
|
||||||
);
|
|
||||||
Err(ErrorKind::GenericError("keybase api fail".to_owned()).into())
|
|
||||||
} else {
|
|
||||||
let response: Value =
|
|
||||||
from_str(from_utf8(&output.stdout).expect("Bad output")).expect("Bad output");
|
|
||||||
let err_msg = format!("{}", response["error"]["message"]);
|
|
||||||
if !err_msg.is_empty() && err_msg != "null" {
|
|
||||||
error!("api_send got error: {}", err_msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(response)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Get keybase username
|
|
||||||
fn whoami() -> Result<String, Error> {
|
|
||||||
let mut proc = Command::new("keybase");
|
|
||||||
proc.args(&["status", "-json"]);
|
|
||||||
let output = proc.output().expect("No output");
|
|
||||||
if !output.status.success() {
|
|
||||||
error!(
|
|
||||||
"keybase api fail: {} {}",
|
|
||||||
String::from_utf8_lossy(&output.stdout),
|
|
||||||
String::from_utf8_lossy(&output.stderr)
|
|
||||||
);
|
|
||||||
Err(ErrorKind::GenericError("keybase api fail".to_owned()).into())
|
|
||||||
} else {
|
|
||||||
let response: Value =
|
|
||||||
from_str(from_utf8(&output.stdout).expect("Bad output")).expect("Bad output");
|
|
||||||
let err_msg = format!("{}", response["error"]["message"]);
|
|
||||||
if !err_msg.is_empty() && err_msg != "null" {
|
|
||||||
error!("status query got error: {}", err_msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
let username = response["Username"].as_str();
|
|
||||||
if let Some(s) = username {
|
|
||||||
Ok(s.to_string())
|
|
||||||
} else {
|
|
||||||
error!("keybase username query fail");
|
|
||||||
Err(ErrorKind::GenericError("keybase username query fail".to_owned()).into())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Get all unread messages from a specific channel/topic and mark as read.
|
|
||||||
fn read_from_channel(channel: &str, topic: &str) -> Result<Vec<String>, Error> {
|
|
||||||
let payload = to_string(&json!({
|
|
||||||
"method": "read",
|
|
||||||
"params": {
|
|
||||||
"options": {
|
|
||||||
"channel": {
|
|
||||||
"name": channel, "topic_type": "dev", "topic_name": topic
|
|
||||||
},
|
|
||||||
"unread_only": true, "peek": false
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
))
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
let response = api_send(&payload);
|
|
||||||
if let Ok(res) = response {
|
|
||||||
let mut unread: Vec<String> = Vec::new();
|
|
||||||
for msg in res["result"]["messages"]
|
|
||||||
.as_array()
|
|
||||||
.unwrap_or(&vec![json!({})])
|
|
||||||
.iter()
|
|
||||||
{
|
|
||||||
if (msg["msg"]["content"]["type"] == "text") && (msg["msg"]["unread"] == true) {
|
|
||||||
let message = msg["msg"]["content"]["text"]["body"].as_str().unwrap_or("");
|
|
||||||
unread.push(message.to_owned());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(unread)
|
|
||||||
} else {
|
|
||||||
Err(ErrorKind::GenericError("keybase api fail".to_owned()).into())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Get unread messages from all channels and mark as read.
|
|
||||||
fn get_unread(topic: &str) -> Result<HashMap<String, String>, Error> {
|
|
||||||
let payload = to_string(&json!({
|
|
||||||
"method": "list",
|
|
||||||
"params": {
|
|
||||||
"options": {
|
|
||||||
"topic_type": "dev",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
.unwrap();
|
|
||||||
let response = api_send(&payload);
|
|
||||||
|
|
||||||
if let Ok(res) = response {
|
|
||||||
let mut channels = HashSet::new();
|
|
||||||
// Unfortunately the response does not contain the message body
|
|
||||||
// and a separate call is needed for each channel
|
|
||||||
for msg in res["result"]["conversations"]
|
|
||||||
.as_array()
|
|
||||||
.unwrap_or(&vec![json!({})])
|
|
||||||
.iter()
|
|
||||||
{
|
|
||||||
if (msg["unread"] == true) && (msg["channel"]["topic_name"] == topic) {
|
|
||||||
let channel = msg["channel"]["name"].as_str().unwrap();
|
|
||||||
channels.insert(channel.to_string());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
let mut unread: HashMap<String, String> = HashMap::new();
|
|
||||||
for channel in channels.iter() {
|
|
||||||
let messages = read_from_channel(channel, topic);
|
|
||||||
if messages.is_err() {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
for msg in messages.unwrap() {
|
|
||||||
unread.insert(msg, channel.to_string());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(unread)
|
|
||||||
} else {
|
|
||||||
Err(ErrorKind::GenericError("keybase api fail".to_owned()).into())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Send a message to a keybase channel that self-destructs after ttl seconds.
|
|
||||||
fn send<T: Serialize>(message: T, channel: &str, topic: &str, ttl: u16) -> bool {
|
|
||||||
let seconds = format!("{}s", ttl);
|
|
||||||
let serialized = to_string(&message).unwrap();
|
|
||||||
let payload = to_string(&json!({
|
|
||||||
"method": "send",
|
|
||||||
"params": {
|
|
||||||
"options": {
|
|
||||||
"channel": {
|
|
||||||
"name": channel, "topic_name": topic, "topic_type": "dev"
|
|
||||||
},
|
|
||||||
"message": {
|
|
||||||
"body": serialized
|
|
||||||
},
|
|
||||||
"exploding_lifetime": seconds
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
.unwrap();
|
|
||||||
let response = api_send(&payload);
|
|
||||||
if let Ok(res) = response {
|
|
||||||
match res["result"]["message"].as_str() {
|
|
||||||
Some("message sent") => {
|
|
||||||
debug!("Message sent to {}: {}", channel, serialized);
|
|
||||||
true
|
|
||||||
}
|
|
||||||
_ => false,
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Send a notify to self that self-destructs after ttl minutes.
|
|
||||||
fn notify(message: &str, channel: &str, ttl: u16) -> bool {
|
|
||||||
let minutes = format!("{}m", ttl);
|
|
||||||
let payload = to_string(&json!({
|
|
||||||
"method": "send",
|
|
||||||
"params": {
|
|
||||||
"options": {
|
|
||||||
"channel": {
|
|
||||||
"name": channel
|
|
||||||
},
|
|
||||||
"message": {
|
|
||||||
"body": message
|
|
||||||
},
|
|
||||||
"exploding_lifetime": minutes
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
.unwrap();
|
|
||||||
let response = api_send(&payload);
|
|
||||||
if let Ok(res) = response {
|
|
||||||
match res["result"]["message"].as_str() {
|
|
||||||
Some("message sent") => true,
|
|
||||||
_ => false,
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Listen for a message from a specific channel with topic SLATE_SIGNED for nseconds and return the first valid slate.
|
|
||||||
fn poll(nseconds: u64, channel: &str) -> Option<Slate> {
|
|
||||||
let start = Instant::now();
|
|
||||||
info!("Waiting for response message from @{}...", channel);
|
|
||||||
while start.elapsed().as_secs() < nseconds {
|
|
||||||
let unread = read_from_channel(channel, SLATE_SIGNED);
|
|
||||||
for msg in unread.unwrap().iter() {
|
|
||||||
let blob = Slate::deserialize_upgrade(&msg);
|
|
||||||
if let Ok(slate) = blob {
|
|
||||||
info!(
|
|
||||||
"keybase response message received from @{}, tx uuid: {}",
|
|
||||||
channel, slate.id,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
sleep(POLL_SLEEP_DURATION);
|
|
||||||
}
|
|
||||||
error!(
|
|
||||||
"No response from @{} in {} seconds. Grin send failed!",
|
|
||||||
channel, nseconds
|
|
||||||
);
|
|
||||||
None
|
|
||||||
}
|
|
||||||
|
|
||||||
impl SlateSender for KeybaseChannel {
|
|
||||||
/// Send a slate to a keybase username then wait for a response for TTL seconds.
|
|
||||||
fn send_tx(&mut self, slate: &Slate, _finalize: bool) -> Result<Slate, Error> {
|
|
||||||
let id = slate.id;
|
|
||||||
|
|
||||||
// Send original slate to recipient with the SLATE_NEW topic
|
|
||||||
match send(&slate, &self.0, SLATE_NEW, TTL) {
|
|
||||||
true => (),
|
|
||||||
false => {
|
|
||||||
return Err(
|
|
||||||
ErrorKind::ClientCallback("Posting transaction slate".to_owned()).into(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
info!("tx request has been sent to @{}, tx uuid: {}", &self.0, id);
|
|
||||||
// Wait for response from recipient with SLATE_SIGNED topic
|
|
||||||
match poll(TTL as u64, &self.0) {
|
|
||||||
Some(slate) => Ok(slate),
|
|
||||||
None => {
|
|
||||||
Err(ErrorKind::ClientCallback("Receiving reply from recipient".to_owned()).into())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Receives slates on all channels with topic SLATE_NEW
|
|
||||||
pub struct KeybaseAllChannels {
|
|
||||||
_priv: (), // makes KeybaseAllChannels unconstructable without checking for existence of keybase executable
|
|
||||||
}
|
|
||||||
|
|
||||||
impl KeybaseAllChannels {
|
|
||||||
/// Create a KeybaseAllChannels, return error if keybase executable is not present
|
|
||||||
pub fn new() -> Result<KeybaseAllChannels, Error> {
|
|
||||||
if !keybase_installed() {
|
|
||||||
Err(ErrorKind::GenericError(
|
|
||||||
"Keybase executable not found, make sure it is installed and in your PATH"
|
|
||||||
.to_owned(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
} else {
|
|
||||||
Ok(KeybaseAllChannels { _priv: () })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl SlateReceiver for KeybaseAllChannels {
|
|
||||||
/// Start a listener, passing received messages to the wallet api directly
|
|
||||||
#[allow(unreachable_code)]
|
|
||||||
fn listen(
|
|
||||||
&self,
|
|
||||||
config: WalletConfig,
|
|
||||||
passphrase: ZeroingString,
|
|
||||||
account: &str,
|
|
||||||
node_api_secret: Option<String>,
|
|
||||||
) -> Result<(), Error> {
|
|
||||||
let node_client = HTTPNodeClient::new(&config.check_node_api_http_addr, node_api_secret)?;
|
|
||||||
let mut wallet =
|
|
||||||
Box::new(DefaultWalletImpl::<'static, HTTPNodeClient>::new(node_client).unwrap())
|
|
||||||
as Box<
|
|
||||||
dyn WalletInst<
|
|
||||||
'static,
|
|
||||||
DefaultLCProvider<HTTPNodeClient, ExtKeychain>,
|
|
||||||
HTTPNodeClient,
|
|
||||||
ExtKeychain,
|
|
||||||
>,
|
|
||||||
>;
|
|
||||||
let lc = wallet.lc_provider().unwrap();
|
|
||||||
lc.set_top_level_directory(&config.data_file_dir)?;
|
|
||||||
let mask = lc.open_wallet(None, passphrase, true, false)?;
|
|
||||||
let wallet_inst = lc.wallet_inst()?;
|
|
||||||
wallet_inst.set_parent_key_id_by_name(account)?;
|
|
||||||
|
|
||||||
info!("Listening for transactions on keybase ...");
|
|
||||||
loop {
|
|
||||||
// listen for messages from all channels with topic SLATE_NEW
|
|
||||||
let unread = get_unread(SLATE_NEW);
|
|
||||||
if unread.is_err() {
|
|
||||||
error!("Listening exited for some keybase api failure");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
for (msg, channel) in &unread.unwrap() {
|
|
||||||
let blob = Slate::deserialize_upgrade(&msg);
|
|
||||||
match blob {
|
|
||||||
Ok(slate) => {
|
|
||||||
let tx_uuid = slate.id;
|
|
||||||
|
|
||||||
// Reject multiple recipients channel for safety
|
|
||||||
{
|
|
||||||
if channel.matches(',').count() > 1 {
|
|
||||||
error!(
|
|
||||||
"Incoming tx initiated on channel \"{}\" is rejected, multiple recipients channel! amount: {}(g), tx uuid: {}",
|
|
||||||
channel,
|
|
||||||
slate.amount as f64 / 1_000_000_000.0,
|
|
||||||
tx_uuid,
|
|
||||||
);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
info!(
|
|
||||||
"tx initiated on channel \"{}\", to send you {}(g). tx uuid: {}",
|
|
||||||
channel,
|
|
||||||
slate.amount as f64 / 1_000_000_000.0,
|
|
||||||
tx_uuid,
|
|
||||||
);
|
|
||||||
let res = {
|
|
||||||
foreign::receive_tx(
|
|
||||||
&mut **wallet_inst,
|
|
||||||
Some(mask.as_ref().unwrap()),
|
|
||||||
&slate,
|
|
||||||
None,
|
|
||||||
false,
|
|
||||||
)
|
|
||||||
};
|
|
||||||
match res {
|
|
||||||
// Reply to the same channel with topic SLATE_SIGNED
|
|
||||||
Ok(s) => {
|
|
||||||
let success = send(s, channel, SLATE_SIGNED, TTL);
|
|
||||||
|
|
||||||
if success {
|
|
||||||
notify_on_receive(
|
|
||||||
config.keybase_notify_ttl.unwrap_or(1440),
|
|
||||||
channel.to_string(),
|
|
||||||
tx_uuid.to_string(),
|
|
||||||
);
|
|
||||||
debug!("Returned slate to @{} via keybase", channel);
|
|
||||||
} else {
|
|
||||||
error!("Failed to return slate to @{} via keybase. Incoming tx failed", channel);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Err(e) => {
|
|
||||||
error!(
|
|
||||||
"Error on receiving tx via keybase: {}. Incoming tx failed",
|
|
||||||
e
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Err(_) => debug!("Failed to deserialize keybase message: {}", msg),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
sleep(LISTEN_SLEEP_DURATION);
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Notify in keybase on receiving a transaction
|
|
||||||
fn notify_on_receive(keybase_notify_ttl: u16, channel: String, tx_uuid: String) {
|
|
||||||
if keybase_notify_ttl > 0 {
|
|
||||||
let my_username = whoami();
|
|
||||||
if let Ok(username) = my_username {
|
|
||||||
let split = channel.split(',');
|
|
||||||
let vec: Vec<&str> = split.collect();
|
|
||||||
if vec.len() > 1 {
|
|
||||||
let receiver = username;
|
|
||||||
let sender = if vec[0] == receiver {
|
|
||||||
vec[1]
|
|
||||||
} else {
|
|
||||||
if vec[1] != receiver {
|
|
||||||
error!("keybase - channel doesn't include my username! channel: {}, username: {}",
|
|
||||||
channel, receiver
|
|
||||||
);
|
|
||||||
}
|
|
||||||
vec[0]
|
|
||||||
};
|
|
||||||
|
|
||||||
let msg = format!(
|
|
||||||
"[grin wallet notice]: \
|
|
||||||
you could have some coins received from @{}\n\
|
|
||||||
Transaction Id: {}",
|
|
||||||
sender, tx_uuid
|
|
||||||
);
|
|
||||||
notify(&msg, &receiver, keybase_notify_ttl);
|
|
||||||
info!(
|
|
||||||
"tx from @{} is done, please check on grin wallet. tx uuid: {}",
|
|
||||||
sender, tx_uuid,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
error!("keybase notification fail on whoami query");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -14,12 +14,10 @@
|
||||||
|
|
||||||
mod file;
|
mod file;
|
||||||
pub mod http;
|
pub mod http;
|
||||||
mod keybase;
|
|
||||||
mod slatepack;
|
mod slatepack;
|
||||||
|
|
||||||
pub use self::file::PathToSlate;
|
pub use self::file::PathToSlate;
|
||||||
pub use self::http::{HttpSlateSender, SchemeNotHttp};
|
pub use self::http::{HttpSlateSender, SchemeNotHttp};
|
||||||
pub use self::keybase::{KeybaseAllChannels, KeybaseChannel};
|
|
||||||
pub use self::slatepack::PathToSlatepack;
|
pub use self::slatepack::PathToSlatepack;
|
||||||
|
|
||||||
use crate::config::WalletConfig;
|
use crate::config::WalletConfig;
|
||||||
|
|
|
@ -44,8 +44,8 @@ pub mod test_framework;
|
||||||
pub mod tor;
|
pub mod tor;
|
||||||
|
|
||||||
pub use crate::adapters::{
|
pub use crate::adapters::{
|
||||||
HttpSlateSender, KeybaseAllChannels, KeybaseChannel, PathToSlate, PathToSlatepack, SlateGetter,
|
HttpSlateSender, PathToSlate, PathToSlatepack, SlateGetter, SlatePutter, SlateReceiver,
|
||||||
SlatePutter, SlateReceiver, SlateSender,
|
SlateSender,
|
||||||
};
|
};
|
||||||
pub use crate::backends::{wallet_db_exists, LMDBBackend};
|
pub use crate::backends::{wallet_db_exists, LMDBBackend};
|
||||||
pub use crate::error::{Error, ErrorKind};
|
pub use crate::error::{Error, ErrorKind};
|
||||||
|
|
|
@ -74,15 +74,6 @@ subcommands:
|
||||||
short: l
|
short: l
|
||||||
long: port
|
long: port
|
||||||
takes_value: true
|
takes_value: true
|
||||||
- method:
|
|
||||||
help: Which method to use for communication
|
|
||||||
short: m
|
|
||||||
long: method
|
|
||||||
possible_values:
|
|
||||||
- http
|
|
||||||
- keybase
|
|
||||||
default_value: http
|
|
||||||
takes_value: true
|
|
||||||
- no_tor:
|
- no_tor:
|
||||||
help: Don't start Tor listener when starting HTTP listener
|
help: Don't start Tor listener when starting HTTP listener
|
||||||
short: n
|
short: n
|
||||||
|
|
Loading…
Reference in a new issue