#![forbid(unsafe_code)]
use super::*;
pub fn append_auth_phase_one(
out: &mut Vec<u8>,
user: &str,
program: &str,
machine: &str,
osuser: &str,
terminal: &str,
pid: u32,
) -> Result<()> {
let mut writer = TtcWriter::new();
writer.write_function_code(TNS_FUNC_AUTH_PHASE_ONE);
write_auth_header(&mut writer, user, TNS_AUTH_MODE_LOGON, 5)?;
write_key_value(&mut writer, "AUTH_TERMINAL", terminal, 0)?;
write_key_value(&mut writer, "AUTH_PROGRAM_NM", program, 0)?;
write_key_value(&mut writer, "AUTH_MACHINE", machine, 0)?;
write_key_value(&mut writer, "AUTH_PID", &pid.to_string(), 0)?;
write_key_value(&mut writer, "AUTH_SID", osuser, 0)?;
out.extend_from_slice(&writer.into_bytes());
Ok(())
}
pub fn build_auth_phase_two_payload(
user: &str,
encrypted: &crate::crypto::EncryptedPassword,
driver_name: &str,
version_num: u32,
connect_string: &str,
) -> Result<Vec<u8>> {
build_auth_phase_two_payload_with_seq(
user,
encrypted,
driver_name,
version_num,
connect_string,
1,
)
}
pub fn build_auth_phase_two_payload_with_seq(
user: &str,
encrypted: &crate::crypto::EncryptedPassword,
driver_name: &str,
version_num: u32,
connect_string: &str,
seq_num: u8,
) -> Result<Vec<u8>> {
build_auth_phase_two_payload_with_context_with_seq(
user,
encrypted,
driver_name,
version_num,
connect_string,
seq_num,
&[],
)
}
pub fn build_auth_phase_two_payload_with_context_with_seq(
user: &str,
encrypted: &crate::crypto::EncryptedPassword,
driver_name: &str,
version_num: u32,
connect_string: &str,
seq_num: u8,
app_context: &[(String, String, String)],
) -> Result<Vec<u8>> {
build_auth_phase_two_payload_with_proxy_with_seq(
user,
encrypted,
driver_name,
version_num,
connect_string,
seq_num,
app_context,
None,
)
}
#[allow(clippy::too_many_arguments)]
pub fn build_auth_phase_two_payload_with_proxy_with_seq(
user: &str,
encrypted: &crate::crypto::EncryptedPassword,
driver_name: &str,
version_num: u32,
connect_string: &str,
seq_num: u8,
app_context: &[(String, String, String)],
proxy_user: Option<&str>,
) -> Result<Vec<u8>> {
let mut writer = TtcWriter::new();
writer.write_function_code_with_seq(TNS_FUNC_AUTH_PHASE_TWO, seq_num);
writer.write_ub8(0);
let mut num_pairs = 6u32;
if encrypted.speedy_key.is_some() {
num_pairs += 1;
}
if proxy_user.is_some() {
num_pairs += 1;
}
if !connect_string.is_empty() {
num_pairs += 1;
}
let app_context_pairs =
app_context
.len()
.checked_mul(3)
.ok_or(ProtocolError::InvalidPacketLength {
length: app_context.len(),
minimum: 0,
})?;
num_pairs +=
u32::try_from(app_context_pairs).map_err(|_| ProtocolError::InvalidPacketLength {
length: app_context.len(),
minimum: 0,
})?;
write_auth_header(
&mut writer,
user,
TNS_AUTH_MODE_LOGON | TNS_AUTH_MODE_WITH_PASSWORD,
num_pairs,
)?;
if let Some(proxy_user) = proxy_user {
write_key_value(&mut writer, "PROXY_CLIENT_NAME", proxy_user, 0)?;
}
write_key_value(&mut writer, "AUTH_SESSKEY", &encrypted.session_key, 1)?;
if let Some(speedy_key) = &encrypted.speedy_key {
write_key_value(&mut writer, "AUTH_PBKDF2_SPEEDY_KEY", speedy_key, 0)?;
}
write_key_value(&mut writer, "AUTH_PASSWORD", &encrypted.password, 0)?;
write_key_value(&mut writer, "SESSION_CLIENT_CHARSET", "873", 0)?;
write_key_value(&mut writer, "SESSION_CLIENT_DRIVER_NAME", driver_name, 0)?;
write_key_value(
&mut writer,
"SESSION_CLIENT_VERSION",
&version_num.to_string(),
0,
)?;
write_key_value(
&mut writer,
"AUTH_ALTER_SESSION",
"ALTER SESSION SET TIME_ZONE='+00:00'\0",
1,
)?;
for (namespace, name, value) in app_context {
write_key_value(&mut writer, "AUTH_APPCTX_NSPACE\0", namespace, 0)?;
write_key_value(&mut writer, "AUTH_APPCTX_ATTR\0", name, 0)?;
write_key_value(&mut writer, "AUTH_APPCTX_VALUE\0", value, 0)?;
}
if !connect_string.is_empty() {
write_key_value(&mut writer, "AUTH_CONNECT_STRING", connect_string, 0)?;
}
Ok(writer.into_bytes())
}
pub fn build_change_password_payload_with_seq(
user: &str,
encoded_password: &str,
encoded_newpassword: &str,
seq_num: u8,
) -> Result<Vec<u8>> {
let mut writer = TtcWriter::new();
writer.write_function_code_with_seq(TNS_FUNC_AUTH_PHASE_TWO, seq_num);
writer.write_ub8(0);
write_auth_header(
&mut writer,
user,
TNS_AUTH_MODE_WITH_PASSWORD | TNS_AUTH_MODE_CHANGE_PASSWORD,
2,
)?;
write_key_value(&mut writer, "AUTH_PASSWORD", encoded_password, 0)?;
write_key_value(&mut writer, "AUTH_NEWPASSWORD", encoded_newpassword, 0)?;
Ok(writer.into_bytes())
}
pub fn parse_auth_response(payload: &[u8]) -> Result<AuthResponse> {
let mut reader = TtcReader::new(payload);
let mut response = AuthResponse::default();
while reader.remaining() > 0 {
let message_type = reader.read_u8()?;
match message_type {
TNS_MSG_TYPE_PROTOCOL => {
if let Some(capabilities) = skip_protocol_message(&mut reader)? {
response.capabilities = Some(capabilities);
}
}
TNS_MSG_TYPE_DATA_TYPES => skip_data_types_response(&mut reader)?,
TNS_MSG_TYPE_PARAMETER => {
let mut parsed = parse_return_parameters(&mut reader)?;
response.session_data.append(&mut parsed.session_data);
if parsed.verifier_type.is_some() {
response.verifier_type = parsed.verifier_type;
}
}
TNS_MSG_TYPE_STATUS => {
let _call_status = reader.read_ub4()?;
let _seq = reader.read_ub2()?;
}
TNS_MSG_TYPE_SERVER_SIDE_PIGGYBACK => {
let _ = skip_server_side_piggyback(&mut reader)?;
}
TNS_MSG_TYPE_END_OF_RESPONSE => break,
TNS_MSG_TYPE_ERROR => {
if let Some(message) = parse_server_error(&mut reader, 13)? {
return Err(ProtocolError::ServerError(message));
}
}
_ => {
return Err(ProtocolError::UnknownMessageType {
message_type,
position: reader.position().saturating_sub(1),
})
}
}
}
Ok(response)
}
pub(crate) fn write_auth_header(
writer: &mut TtcWriter,
user: &str,
auth_mode: u32,
num_pairs: u32,
) -> Result<()> {
let user_bytes = user.as_bytes();
writer.write_u8(u8::from(!user_bytes.is_empty()));
writer.write_ub4(u32::try_from(user_bytes.len()).map_err(|_| {
ProtocolError::InvalidPacketLength {
length: user_bytes.len(),
minimum: 0,
}
})?);
writer.write_ub4(auth_mode);
writer.write_u8(1);
writer.write_ub4(num_pairs);
writer.write_u8(1);
writer.write_u8(1);
if !user_bytes.is_empty() {
writer.write_bytes_with_length(user_bytes)?;
}
Ok(())
}
pub(crate) fn write_key_value(
writer: &mut TtcWriter,
key: &str,
value: &str,
flags: u32,
) -> Result<()> {
writer.write_str_two_lengths(key)?;
writer.write_str_two_lengths(value)?;
writer.write_ub4(flags);
Ok(())
}
pub(crate) fn parse_return_parameters(reader: &mut TtcReader<'_>) -> Result<AuthResponse> {
let num_params = reader.read_ub2()?;
let mut response = AuthResponse::default();
for _ in 0..num_params {
let key = reader
.read_string_with_length()?
.ok_or(ProtocolError::TtcDecode("missing auth response key"))?;
let value = reader.read_string_with_length()?.unwrap_or_default();
if key == "AUTH_VFR_DATA" {
response.verifier_type = Some(reader.read_ub4()?);
} else {
let _flags = reader.read_ub4()?;
}
response.session_data.insert(key, value);
}
Ok(response)
}