Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add WASM support #7

Draft
wants to merge 4 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -16,3 +16,6 @@ sha-1 = "0.10.0"
sha2 = "0.10.1"
base32 = "0.4.0"
url = "2.2.2"

[target.'cfg(target_arch = "wasm32")'.dependencies]
wasm-bindgen = "0.2"
7 changes: 7 additions & 0 deletions src/hotp.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,9 @@
use crate::otp_result::OTPResult;
use crate::util::{base32_decode, get_code, hash_generic, MacDigest};

#[cfg(target_arch = "wasm32")]
use wasm_bindgen::prelude::*;

/// A HOTP Generator
///
/// Follows the specification listed in [RFC4226]. Needs a secret and
Expand All @@ -18,6 +21,7 @@ use crate::util::{base32_decode, get_code, hash_generic, MacDigest};
///
/// [RFC4226]: https://datatracker.ietf.org/doc/html/rfc4226

#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
#[derive(Debug, Clone, Hash)]
pub struct HOTP {
/// The secret key used in the HMAC process.
Expand All @@ -33,6 +37,7 @@ pub struct HOTP {
}

/// All initializer implementations for the [`HOTP`] struct.
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl HOTP {
/// Creates a new HOTP instance with a byte-array representation
/// of the secret and specified digit count.
Expand Down Expand Up @@ -87,6 +92,7 @@ impl HOTP {
}

/// All getters for the ['HOTP'] struct
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl HOTP {
/// Gets the number of digits of the code.
pub fn get_digits(&self) -> u32 {
Expand All @@ -95,6 +101,7 @@ impl HOTP {
}

/// All otp generation methods for the [`HOTP`] struct.
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl HOTP {
/// Generates and returns the HOTP value.
///
Expand Down
5 changes: 4 additions & 1 deletion src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -80,4 +80,7 @@
pub mod hotp;
pub mod totp;
pub mod util;
pub mod otp_result;
pub mod otp_result;

#[cfg(target_arch = "wasm32")]
pub mod wasm_utils;
7 changes: 7 additions & 0 deletions src/otp_result.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,9 @@
use std::fmt;
use std::fmt::Formatter;

#[cfg(target_arch = "wasm32")]
use wasm_bindgen::prelude::*;

/// A convenience struct to hold the result of a [`HOTP`] or [`TOTP`]
/// generation.
///
Expand All @@ -13,12 +16,14 @@ use std::fmt::Formatter;
/// Returned as a result of either [`HOTP::get_otp`], [`TOTP::get_otp`]
/// or [`TOTP::get_otp_with_custom_time_start`].
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq)]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
pub struct OTPResult {
digits: u32,
code: u32,
}

/// Constructors for the [`OTPResult`] struct.
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl OTPResult {
/// Creates a new instance with the provided digit count and OTP code.
pub fn new(digits: u32, code: u32 ) -> Self {
Expand All @@ -27,6 +32,7 @@ impl OTPResult {
}

/// Getters for the [`OTPResult`] struct.
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl OTPResult {
/// Gets the digit count given to the struct on creation.
///
Expand All @@ -35,6 +41,7 @@ impl OTPResult {
}

/// Convenience code getters for the [`OTPResult`] struct
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl OTPResult {
/// Returns the OTP as a formatted string of length [`OTPResult.digits`].
///
Expand Down
8 changes: 8 additions & 0 deletions src/totp.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,9 @@
use crate::otp_result::OTPResult;
use crate::util::{base32_decode, get_code, hash_generic, MacDigest};

#[cfg(target_arch = "wasm32")]
use wasm_bindgen::prelude::*;

/// A TOTP generator
///
/// Follows the specification listed in [RFC6238]. Needs a secret,
Expand All @@ -16,6 +19,7 @@ use crate::util::{base32_decode, get_code, hash_generic, MacDigest};
/// utilized in a similar manner.
///
/// [RFC6238]: https://datatracker.ietf.org/doc/html/rfc6238
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
#[derive(Debug, Clone, Hash)]
pub struct TOTP {
/// The secret key used in the HMAC process.
Expand All @@ -42,6 +46,7 @@ pub struct TOTP {
}

/// All initializer implementations for the [`TOTP`] struct
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl TOTP {
/// Generates a new TOTP instance from a byte array representation of the
/// secret, a digest algorithm, a number of digits,
Expand Down Expand Up @@ -132,6 +137,7 @@ impl TOTP {
}

/// All getters for the [`TOTP`] struct
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl TOTP {
/// Gets the algorithm used for code generation.
pub fn get_digest(&self) -> MacDigest {
Expand All @@ -150,6 +156,7 @@ impl TOTP {
}

/// All helper methods for totp generation
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl TOTP {

/// Returns the time in seconds until an OTP refresh is needed.
Expand All @@ -173,6 +180,7 @@ impl TOTP {
}

/// All otp generation methods for the [`TOTP`] struct.
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
impl TOTP {
/// Generates and returns the TOTP value for the specified time.
///
Expand Down
6 changes: 5 additions & 1 deletion src/util.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,9 @@ use url::Url;
use crate::hotp::HOTP;
use crate::totp::TOTP;

#[cfg(target_arch = "wasm32")]
use wasm_bindgen::prelude::*;

/// The digest to use with TOTP.
///
/// All three digests referenced in [RFC6238] are supported:
Expand All @@ -20,6 +23,7 @@ use crate::totp::TOTP;
///
/// [RFC6238]: https://datatracker.ietf.org/doc/html/rfc6238
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq)]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
pub enum MacDigest {
SHA1,
SHA256,
Expand Down Expand Up @@ -194,4 +198,4 @@ pub fn parse_otpauth_uri(uri: &str) -> Result<ParseResult, ParseError> {
} else {
Err(UnknownOtpType(String::from(type_str)))
}
}
}
85 changes: 85 additions & 0 deletions src/wasm_utils.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
use crate::hotp::HOTP;
use crate::totp::TOTP;
use crate::util::parse_otpauth_uri;
use crate::util;

use wasm_bindgen::prelude::*;

// WASM-specific code (Non C-style enums aren't supported by wasm_bindgen)
// So we have to add some compatibility to the parse_otpauth_uri method

/// A tuple struct to hold an OTP Parse Result
///
/// Holds an [`Option`] of each HOTP (with counter) or TOTP parse result.
/// The value which isn't of type [`None`] is the parse result. If both
/// are [`None`] then there was an error in parsing the URI.
#[wasm_bindgen]
pub struct ParseResult(
/// The potential HOTP result given by the parser
Option<HOTPResult>,
/// The potential TOTP result given by the parser
Option<TOTP>,
);

/// Getters for the [`ParseResult`] struct
#[wasm_bindgen]
impl ParseResult {
/// Gets the [`HOTPResult`] provided by the parser
///
/// If [`None`], then the parsed URI was not a HOTP URI.
/// If a value, then the parsed URI was HOTP and also contains the
/// associated counter.
#[wasm_bindgen(getter)]
pub fn get_hotp_result(&self) -> Option<HOTPResult> {
self.0.clone()
}

/// Gets the [`TOTP`] provided by the parser
///
/// If [`None`], then the parsed URI was not a TOTP URI.
/// If a value, then the parsed URI was TOTP and OTPs can be generated
#[wasm_bindgen(getter)]
pub fn get_totp(&self) -> Option<TOTP> {
self.1.clone()
}
}

/// A tuple struct to hold the HOTP result given by a parser
///
/// Holds an instance of the [`HOTP`] struct and a numeric counter
#[wasm_bindgen]
#[derive(Clone)]
pub struct HOTPResult(
/// The HOTP instance
HOTP,
/// The counter needed for OTP generation
u64
);

/// Getters for the [`HOTPResult`] struct
#[wasm_bindgen]
impl HOTPResult {
/// Gets the [`HOTP`] instance associated with this result
#[wasm_bindgen(getter)]
pub fn get_hotp(&self) -> HOTP {
self.0.clone()
}

/// Gets the current counter value for use with the HOTP generation
#[wasm_bindgen(getter)]
pub fn get_counter(&self) -> u64 {
self.1
}
}

/// A wasm-compatible method to parse an otpauth URI into its specific OTP
/// generator. Returns the [`ParseResult`] object, which will contain
/// one or neither of the HOTP/TOTP instances.
#[wasm_bindgen]
pub fn parse_otpauth_uri_wasm(uri: &str) -> ParseResult {
match parse_otpauth_uri(uri) {
Ok(util::ParseResult::HOTP(result, counter)) => ParseResult(Some(HOTPResult(result, counter)), None),
Ok(util::ParseResult::TOTP(result)) => ParseResult(None, Some(result)),
Err(_e) => ParseResult(None, None)
}
}