1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
use std::sync::Arc;

#[cfg(unix)]
use std::os::unix::io::AsRawFd;
#[cfg(windows)]
use std::os::windows::io::{AsRawSocket, BorrowedSocket};

use ssh2::{Agent, PublicKey, Session};

use crate::{error::Error, session::get_session, session_stream::AsyncSessionStream};

//
pub struct AsyncAgent<S> {
    inner: Agent,
    sess: Session,
    stream: Arc<S>,
}

#[cfg(unix)]
impl<S> AsyncAgent<S>
where
    S: AsRawFd + 'static,
{
    pub fn new(stream: S) -> Result<Self, Error> {
        let mut session = get_session(None)?;
        session.set_tcp_stream(stream.as_raw_fd());

        let stream = Arc::new(stream);

        let agent = session.agent()?;

        Ok(Self {
            inner: agent,
            sess: session,
            stream,
        })
    }
}

#[cfg(windows)]
impl<S> AsyncAgent<S>
where
    S: AsRawSocket + 'static,
{
    pub fn new(stream: S) -> Result<Self, Error> {
        let mut session = get_session(None)?;
        session.set_tcp_stream(unsafe { BorrowedSocket::borrow_raw(stream.as_raw_socket()) });

        let stream = Arc::new(stream);

        let agent = session.agent()?;

        Ok(Self {
            inner: agent,
            sess: session,
            stream,
        })
    }
}

impl<S> AsyncAgent<S> {
    pub(crate) fn from_parts(inner: Agent, sess: Session, stream: Arc<S>) -> Self {
        Self {
            inner,
            sess,
            stream,
        }
    }
}

impl<S> AsyncAgent<S> {
    pub fn identities(&self) -> Result<Vec<PublicKey>, Error> {
        self.inner.identities().map_err(Into::into)
    }
}

impl<S> AsyncAgent<S>
where
    S: AsyncSessionStream + Send + Sync + 'static,
{
    pub async fn connect(&mut self) -> Result<(), Error> {
        self.stream
            .none_with(|| self.inner.connect(), &self.sess)
            .await
    }

    pub async fn disconnect(&mut self) -> Result<(), Error> {
        self.stream
            .none_with(|| self.inner.disconnect(), &self.sess)
            .await
    }

    pub async fn list_identities(&mut self) -> Result<(), Error> {
        self.stream
            .rw_with(|| self.inner.list_identities(), &self.sess)
            .await
    }

    pub async fn userauth(&self, username: &str, identity: &PublicKey) -> Result<(), Error> {
        self.stream
            .rw_with(|| self.inner.userauth(username, identity), &self.sess)
            .await
    }
}