cleaned up channel binding logic, cleaned up SaslCredentials, updated documentation
This commit is contained in:
parent
4a8b405f3b
commit
88ea00baa7
5 changed files with 92 additions and 56 deletions
|
@ -2,7 +2,7 @@
|
||||||
name = "sasl"
|
name = "sasl"
|
||||||
version = "0.1.1"
|
version = "0.1.1"
|
||||||
authors = ["lumi <lumi@pew.im>"]
|
authors = ["lumi <lumi@pew.im>"]
|
||||||
description = "A crate for SASL authentication. Still needs a bunch of documenation."
|
description = "A crate for SASL authentication."
|
||||||
homepage = "https://gitlab.com/lumi/sasl-rs"
|
homepage = "https://gitlab.com/lumi/sasl-rs"
|
||||||
repository = "https://gitlab.com/lumi/sasl-rs"
|
repository = "https://gitlab.com/lumi/sasl-rs"
|
||||||
documentation = "https://docs.rs/sasl"
|
documentation = "https://docs.rs/sasl"
|
||||||
|
|
|
@ -6,6 +6,11 @@ What's this?
|
||||||
|
|
||||||
A crate which handles SASL authentication.
|
A crate which handles SASL authentication.
|
||||||
|
|
||||||
|
Can I see an example?
|
||||||
|
---------------------
|
||||||
|
|
||||||
|
Look at the documentation [here](https://docs.rs/sasl).
|
||||||
|
|
||||||
What license is it under?
|
What license is it under?
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
|
|
|
@ -5,14 +5,12 @@
|
||||||
//! # Examples
|
//! # Examples
|
||||||
//!
|
//!
|
||||||
//! ```rust
|
//! ```rust
|
||||||
//! use sasl::{SaslCredentials, SaslSecret, SaslMechanism, Error};
|
//! use sasl::{SaslCredentials, SaslMechanism, Error};
|
||||||
//! use sasl::mechanisms::Plain;
|
//! use sasl::mechanisms::Plain;
|
||||||
//!
|
//!
|
||||||
//! let creds = SaslCredentials {
|
//! let creds = SaslCredentials::default()
|
||||||
//! username: "user".to_owned(),
|
//! .with_username("user")
|
||||||
//! secret: SaslSecret::Password("pencil".to_owned()),
|
//! .with_password("pencil");
|
||||||
//! channel_binding: None,
|
|
||||||
//! };
|
|
||||||
//!
|
//!
|
||||||
//! let mut mechanism = Plain::from_credentials(creds).unwrap();
|
//! let mut mechanism = Plain::from_credentials(creds).unwrap();
|
||||||
//!
|
//!
|
||||||
|
@ -39,16 +37,75 @@ mod error;
|
||||||
pub use error::Error;
|
pub use error::Error;
|
||||||
|
|
||||||
/// A struct containing SASL credentials.
|
/// A struct containing SASL credentials.
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
pub struct SaslCredentials {
|
pub struct SaslCredentials {
|
||||||
/// The requested username.
|
/// The requested username.
|
||||||
pub username: String, // TODO: change this since some mechanisms do not use it
|
pub username: Option<String>,
|
||||||
/// The secret used to authenticate.
|
/// The secret used to authenticate.
|
||||||
pub secret: SaslSecret,
|
pub secret: SaslSecret,
|
||||||
/// Optionally, channel binding data, for *-PLUS mechanisms.
|
/// Channel binding data, for *-PLUS mechanisms.
|
||||||
pub channel_binding: Option<Vec<u8>>,
|
pub channel_binding: ChannelBinding,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for SaslCredentials {
|
||||||
|
fn default() -> SaslCredentials {
|
||||||
|
SaslCredentials {
|
||||||
|
username: None,
|
||||||
|
secret: SaslSecret::None,
|
||||||
|
channel_binding: ChannelBinding::None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SaslCredentials {
|
||||||
|
/// Creates a new SaslCredentials with the specified username.
|
||||||
|
pub fn with_username<N: Into<String>>(mut self, username: N) -> SaslCredentials {
|
||||||
|
self.username = Some(username.into());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates a new SaslCredentials with the specified password.
|
||||||
|
pub fn with_password<P: Into<String>>(mut self, password: P) -> SaslCredentials {
|
||||||
|
self.secret = SaslSecret::Password(password.into());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates a new SaslCredentials with the specified chanel binding.
|
||||||
|
pub fn with_channel_binding(mut self, channel_binding: ChannelBinding) -> SaslCredentials {
|
||||||
|
self.channel_binding = channel_binding;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Channel binding configuration.
|
||||||
|
#[derive(Clone, Debug, PartialEq, Eq)]
|
||||||
|
pub enum ChannelBinding {
|
||||||
|
/// No channel binding data.
|
||||||
|
None,
|
||||||
|
/// p=tls-unique channel binding data.
|
||||||
|
TlsUnique(Vec<u8>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ChannelBinding {
|
||||||
|
/// Return the gs2 header for this channel binding mechanism.
|
||||||
|
pub fn header(&self) -> &[u8] {
|
||||||
|
match *self {
|
||||||
|
ChannelBinding::None => b"n,,",
|
||||||
|
ChannelBinding::TlsUnique(_) => b"p=tls-unique,,",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return the channel binding data for this channel binding mechanism.
|
||||||
|
pub fn data(&self) -> &[u8] {
|
||||||
|
match *self {
|
||||||
|
ChannelBinding::None => &[],
|
||||||
|
ChannelBinding::TlsUnique(ref data) => data,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Represents a SASL secret, like a password.
|
/// Represents a SASL secret, like a password.
|
||||||
|
#[derive(Clone, Debug, PartialEq, Eq)]
|
||||||
pub enum SaslSecret {
|
pub enum SaslSecret {
|
||||||
/// No extra data needed.
|
/// No extra data needed.
|
||||||
None,
|
None,
|
||||||
|
|
|
@ -30,7 +30,11 @@ impl SaslMechanism for Plain {
|
||||||
|
|
||||||
fn from_credentials(credentials: SaslCredentials) -> Result<Plain, String> {
|
fn from_credentials(credentials: SaslCredentials) -> Result<Plain, String> {
|
||||||
if let SaslSecret::Password(password) = credentials.secret {
|
if let SaslSecret::Password(password) = credentials.secret {
|
||||||
Ok(Plain::new(credentials.username, password))
|
if let Some(username) = credentials.username {
|
||||||
|
Ok(Plain::new(username, password))
|
||||||
|
} else {
|
||||||
|
Err("PLAIN requires a username".to_owned())
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
Err("PLAIN requires a password".to_owned())
|
Err("PLAIN requires a password".to_owned())
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
use base64;
|
use base64;
|
||||||
|
|
||||||
|
use ChannelBinding;
|
||||||
use SaslCredentials;
|
use SaslCredentials;
|
||||||
use SaslMechanism;
|
use SaslMechanism;
|
||||||
use SaslSecret;
|
use SaslSecret;
|
||||||
|
@ -153,18 +154,20 @@ pub struct Scram<S: ScramProvider> {
|
||||||
password: String,
|
password: String,
|
||||||
client_nonce: String,
|
client_nonce: String,
|
||||||
state: ScramState,
|
state: ScramState,
|
||||||
channel_binding: Option<Vec<u8>>,
|
channel_binding: ChannelBinding,
|
||||||
_marker: PhantomData<S>,
|
_marker: PhantomData<S>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S: ScramProvider> Scram<S> {
|
impl<S: ScramProvider> Scram<S> {
|
||||||
/// Constructs a new struct for authenticating using the SASL SCRAM-* mechanism.
|
/// Constructs a new struct for authenticating using the SASL SCRAM-* and SCRAM-*-PLUS
|
||||||
|
/// mechanisms, depending on the passed channel binding.
|
||||||
///
|
///
|
||||||
/// It is recommended that instead you use a `SaslCredentials` struct and turn it into the
|
/// It is recommended that instead you use a `SaslCredentials` struct and turn it into the
|
||||||
/// requested mechanism using `from_credentials`.
|
/// requested mechanism using `from_credentials`.
|
||||||
pub fn new<N: Into<String>, P: Into<String>>(
|
pub fn new<N: Into<String>, P: Into<String>>(
|
||||||
username: N,
|
username: N,
|
||||||
password: P,
|
password: P,
|
||||||
|
channel_binding: ChannelBinding,
|
||||||
) -> Result<Scram<S>, Error> {
|
) -> Result<Scram<S>, Error> {
|
||||||
Ok(Scram {
|
Ok(Scram {
|
||||||
name: format!("SCRAM-{}", S::name()),
|
name: format!("SCRAM-{}", S::name()),
|
||||||
|
@ -172,17 +175,14 @@ impl<S: ScramProvider> Scram<S> {
|
||||||
password: password.into(),
|
password: password.into(),
|
||||||
client_nonce: generate_nonce()?,
|
client_nonce: generate_nonce()?,
|
||||||
state: ScramState::Init,
|
state: ScramState::Init,
|
||||||
channel_binding: None,
|
channel_binding: channel_binding,
|
||||||
_marker: PhantomData,
|
_marker: PhantomData,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Constructs a new struct for authenticating using the SASL SCRAM-* mechanism.
|
// Used for testing.
|
||||||
///
|
#[doc(hidden)]
|
||||||
/// This one takes a nonce instead of generating it.
|
#[cfg(test)]
|
||||||
///
|
|
||||||
/// It is recommended that instead you use a `SaslCredentials` struct and turn it into the
|
|
||||||
/// requested mechanism using `from_credentials`.
|
|
||||||
pub fn new_with_nonce<N: Into<String>, P: Into<String>>(
|
pub fn new_with_nonce<N: Into<String>, P: Into<String>>(
|
||||||
username: N,
|
username: N,
|
||||||
password: P,
|
password: P,
|
||||||
|
@ -194,33 +194,10 @@ impl<S: ScramProvider> Scram<S> {
|
||||||
password: password.into(),
|
password: password.into(),
|
||||||
client_nonce: nonce,
|
client_nonce: nonce,
|
||||||
state: ScramState::Init,
|
state: ScramState::Init,
|
||||||
channel_binding: None,
|
channel_binding: ChannelBinding::None,
|
||||||
_marker: PhantomData,
|
_marker: PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Constructs a new struct for authenticating using the SASL SCRAM-*-PLUS mechanism.
|
|
||||||
///
|
|
||||||
/// This means that this function will also take the channel binding data.
|
|
||||||
///
|
|
||||||
/// It is recommended that instead you use a `SaslCredentials` struct and turn it into the
|
|
||||||
/// requested mechanism using `from_credentials`.
|
|
||||||
pub fn new_with_channel_binding<N: Into<String>, P: Into<String>>(
|
|
||||||
username: N,
|
|
||||||
password: P,
|
|
||||||
channel_binding: Vec<u8>,
|
|
||||||
) -> Result<Scram<S>, Error> {
|
|
||||||
// TODO: channel binding modes other than tls-unique
|
|
||||||
Ok(Scram {
|
|
||||||
name: format!("SCRAM-{}-PLUS", S::name()),
|
|
||||||
username: username.into(),
|
|
||||||
password: password.into(),
|
|
||||||
client_nonce: generate_nonce()?,
|
|
||||||
state: ScramState::Init,
|
|
||||||
channel_binding: Some(channel_binding),
|
|
||||||
_marker: PhantomData,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S: ScramProvider> SaslMechanism for Scram<S> {
|
impl<S: ScramProvider> SaslMechanism for Scram<S> {
|
||||||
|
@ -231,12 +208,11 @@ impl<S: ScramProvider> SaslMechanism for Scram<S> {
|
||||||
|
|
||||||
fn from_credentials(credentials: SaslCredentials) -> Result<Scram<S>, String> {
|
fn from_credentials(credentials: SaslCredentials) -> Result<Scram<S>, String> {
|
||||||
if let SaslSecret::Password(password) = credentials.secret {
|
if let SaslSecret::Password(password) = credentials.secret {
|
||||||
if let Some(binding) = credentials.channel_binding {
|
if let Some(username) = credentials.username {
|
||||||
Scram::new_with_channel_binding(credentials.username, password, binding)
|
Scram::new(username, password, credentials.channel_binding)
|
||||||
.map_err(|_| "can't generate nonce".to_owned())
|
.map_err(|_| "can't generate nonce".to_owned())
|
||||||
} else {
|
} else {
|
||||||
Scram::new(credentials.username, password)
|
Err("SCRAM requires a username".to_owned())
|
||||||
.map_err(|_| "can't generate nonce".to_owned())
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Err("SCRAM requires a password".to_owned())
|
Err("SCRAM requires a password".to_owned())
|
||||||
|
@ -245,11 +221,7 @@ impl<S: ScramProvider> SaslMechanism for Scram<S> {
|
||||||
|
|
||||||
fn initial(&mut self) -> Result<Vec<u8>, String> {
|
fn initial(&mut self) -> Result<Vec<u8>, String> {
|
||||||
let mut gs2_header = Vec::new();
|
let mut gs2_header = Vec::new();
|
||||||
if let Some(_) = self.channel_binding {
|
gs2_header.extend(self.channel_binding.header());
|
||||||
gs2_header.extend(b"p=tls-unique,,");
|
|
||||||
} else {
|
|
||||||
gs2_header.extend(b"n,,");
|
|
||||||
}
|
|
||||||
let mut bare = Vec::new();
|
let mut bare = Vec::new();
|
||||||
bare.extend(b"n=");
|
bare.extend(b"n=");
|
||||||
bare.extend(self.username.bytes());
|
bare.extend(self.username.bytes());
|
||||||
|
@ -286,9 +258,7 @@ impl<S: ScramProvider> SaslMechanism for Scram<S> {
|
||||||
client_final_message_bare.extend(b"c=");
|
client_final_message_bare.extend(b"c=");
|
||||||
let mut cb_data: Vec<u8> = Vec::new();
|
let mut cb_data: Vec<u8> = Vec::new();
|
||||||
cb_data.extend(gs2_header);
|
cb_data.extend(gs2_header);
|
||||||
if let Some(ref cb) = self.channel_binding {
|
cb_data.extend(self.channel_binding.data());
|
||||||
cb_data.extend(cb);
|
|
||||||
}
|
|
||||||
client_final_message_bare.extend(base64::encode(&cb_data).bytes());
|
client_final_message_bare.extend(base64::encode(&cb_data).bytes());
|
||||||
client_final_message_bare.extend(b",r=");
|
client_final_message_bare.extend(b",r=");
|
||||||
client_final_message_bare.extend(server_nonce.bytes());
|
client_final_message_bare.extend(server_nonce.bytes());
|
||||||
|
|
Loading…
Reference in a new issue