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
/*******************************************************************************
*   (c) 2020 Zondax GmbH
*
*  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.
********************************************************************************/
//! Generic APDU transport library for Ledger Nano S/X apps

#![deny(warnings, trivial_casts, trivial_numeric_casts)]
#![deny(unused_import_braces, unused_qualifications)]
#![deny(missing_docs)]

pub use ledger_apdu::{APDUAnswer, APDUCommand, APDUErrorCodes};

/// APDU Errors
pub mod errors;

#[cfg(target_arch = "wasm32")]
/// APDU Transport wrapper for JS/WASM transports
pub mod apdu_transport_wasm;

#[cfg(target_arch = "wasm32")]
pub use crate::apdu_transport_wasm::{APDUTransport, TransportWrapperTrait};

#[cfg(not(target_arch = "wasm32"))]
pub mod apdu_transport_native;

#[cfg(not(target_arch = "wasm32"))]
pub use crate::apdu_transport_native::APDUTransport;

#[cfg(not(target_arch = "wasm32"))]
pub use exchange::Exchange;

#[cfg(not(target_arch = "wasm32"))]
pub mod exchange {
    //! Some implementation on transport typos for the Exchange trait

    use futures::future;
    use trait_async::trait_async;

    use crate::errors::TransportError;

    use ledger_apdu::{APDUAnswer, APDUCommand};

    /// Use to talk to the ledger device
    #[trait_async]
    pub trait Exchange: Send + Sync {
        /// Use to talk to the ledger device
        async fn exchange(&self, command: &APDUCommand) -> Result<APDUAnswer, TransportError>;
    }

    #[trait_async]
    impl Exchange for ledger::TransportNativeHID {
        async fn exchange(&self, command: &APDUCommand) -> Result<APDUAnswer, TransportError> {
            let call = self
                .exchange(command)
                .map_err(|_| TransportError::APDUExchangeError)?;
            future::ready(Ok(call)).await
        }
    }

    #[cfg(feature = "zemu")]
    #[trait_async]
    impl Exchange for ledger_zemu::TransportZemuGrpc {
        async fn exchange(&self, command: &APDUCommand) -> Result<APDUAnswer, TransportError> {
            self.exchange(command)
                .await
                .map_err(|_| TransportError::APDUExchangeError)
        }
    }

    #[cfg(feature = "zemu")]
    #[trait_async]
    impl Exchange for ledger_zemu::TransportZemuHttp {
        async fn exchange(&self, command: &APDUCommand) -> Result<APDUAnswer, TransportError> {
            self.exchange(command)
                .await
                .map_err(|_| TransportError::APDUExchangeError)
        }
    }
}