Skip to content

Commit

Permalink
Merge pull request #10 from sloganking/use-enigo-release-version-again
Browse files Browse the repository at this point in the history
Use enigo release version again
  • Loading branch information
sloganking authored Jan 5, 2024
2 parents 99bccb9 + cc4c95e commit 0414b18
Show file tree
Hide file tree
Showing 2 changed files with 102 additions and 71 deletions.
1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ clap = { version = "4.4.6", features = ["derive"] }
cpal = "0.15.2"
dotenvy = "0.15.7"
enigo = "0.1.3"
flume = "0.11.0"
hound = "3.5.1"
rdev = "0.5.3"
tempfile = "3.8.0"
Expand Down
172 changes: 101 additions & 71 deletions src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ use enigo::{Enigo, KeyboardControllable};
use std::env;
use tempfile::tempdir;
mod transcribe;
use std::thread;
use transcribe::trans;
mod record;
use async_std::future;
Expand Down Expand Up @@ -344,97 +345,126 @@ fn main() -> Result<(), Box<dyn Error>> {
return Ok(());
}

let mut recorder = rec::Recorder::new();
let client = Client::new();
let runtime =
tokio::runtime::Runtime::new().context("Failed to create tokio runtime")?;
let mut enigo = Enigo::new();
let (tx, rx): (flume::Sender<Event>, flume::Receiver<Event>) = flume::unbounded();

let tmp_dir = tempdir()?;
// println!("{:?}", tmp_dir.path());
let voice_tmp_path = tmp_dir.path().join("voice_tmp.wav");
// create key handler thread
thread::spawn(move || {
let mut recorder = rec::Recorder::new();
let client = Client::new();
let runtime = tokio::runtime::Runtime::new()
.context("Failed to create tokio runtime")
.unwrap();
let mut enigo = Enigo::new();

let mut recording_start = std::time::SystemTime::now();
let mut key_pressed = false;
let tmp_dir = tempdir().unwrap();
// println!("{:?}", tmp_dir.path());
let voice_tmp_path = tmp_dir.path().join("voice_tmp.wav");

let callback = move |event: Event| {
let mut recording_start = std::time::SystemTime::now();
let mut key_pressed = false;
let key_to_check = ptt_key;
match event.event_type {
rdev::EventType::KeyPress(key) => {
if key == key_to_check && !key_pressed {
key_pressed = true;
// handle key press
recording_start = std::time::SystemTime::now();
match recorder.start_recording(&voice_tmp_path, Some(&opt.device)) {
Ok(_) => (),
Err(err) => println!("Error: Failed to start recording: {:?}", err),
}
}
}
rdev::EventType::KeyRelease(key) => {
if key == key_to_check && key_pressed {
key_pressed = false;
// handle key release

// get elapsed time since recording started
let elapsed = match recording_start.elapsed() {
Ok(elapsed) => elapsed,
Err(err) => {
println!(
"Error: Failed to get elapsed recording time. Skipping transcription: \n\n{}",err
);
return;
}
};
match recorder.stop_recording() {
Ok(_) => (),
Err(err) => {
println!("Error: Failed to stop recording: {:?}", err);
return;

for event in rx.iter() {
// println!("Received: {:?}", event);
match event.event_type {
rdev::EventType::KeyPress(key) => {
if key == key_to_check && !key_pressed {
key_pressed = true;
// handle key press
recording_start = std::time::SystemTime::now();
match recorder.start_recording(&voice_tmp_path, Some(&opt.device)) {
Ok(_) => (),
Err(err) => {
println!("Error: Failed to start recording: {:?}", err)
}
}
}
}
rdev::EventType::KeyRelease(key) => {
if key == key_to_check && key_pressed {
key_pressed = false;
// handle key release

// Whisper API can't handle less than 0.1 seconds of audio.
// So we'll only transcribe if the recording is longer than 0.2 seconds.
if elapsed.as_secs_f32() > 0.2 {
let transcription_result = match runtime.block_on(future::timeout(
Duration::from_secs(10),
trans::transcribe(&client, &voice_tmp_path),
)) {
Ok(transcription_result) => transcription_result,
// get elapsed time since recording started
let elapsed = match recording_start.elapsed() {
Ok(elapsed) => elapsed,
Err(err) => {
println!("Error: Failed to transcribe audio due to timeout: {:?}", err);
return;
println!("Error: Failed to get elapsed recording time. Skipping transcription: \n\n{}",err);
continue;
}
};

let mut transcription = match transcription_result {
Ok(transcription) => transcription,
match recorder.stop_recording() {
Ok(_) => (),
Err(err) => {
println!("Error: Failed to transcribe audio: {:?}", err);
return;
println!("Error: Failed to stop recording: {:?}", err);
continue;
}
};
}

if let Some(last_char) = transcription.chars().last() {
if ['.', '?', '!', ','].contains(&last_char) {
transcription.push(' ');
// future::timeout(
// Duration::from_secs(10),
// trans::transcribe(&client, &voice_tmp_path),
// )
// .await;

// Whisper API can't handle less than 0.1 seconds of audio.
// So we'll only transcribe if the recording is longer than 0.2 seconds.
if elapsed.as_secs_f32() > 0.2 {
let transcription_result = match runtime.block_on(
future::timeout(
Duration::from_secs(10),
trans::transcribe(&client, &voice_tmp_path),
),
) {
Ok(transcription_result) => transcription_result,
Err(err) => {
println!("Error: Failed to transcribe audio due to timeout: {:?}", err);
continue;
}
};

let mut transcription = match transcription_result {
Ok(transcription) => transcription,
Err(err) => {
println!(
"Error: Failed to transcribe audio: {:?}",
err
);
continue;
}
};

if let Some(last_char) = transcription.chars().last() {
if ['.', '?', '!', ','].contains(&last_char) {
transcription.push(' ');
}
}

if transcription.is_empty() {
println!("No transcription");
}
}

enigo.key_sequence(&transcription);
} else {
println!("Recording too short");
enigo.key_sequence(&transcription);
} else {
println!("Recording too short");
}
}
}
_ => (),
}
_ => (),
}
};
});

// This will block.
if let Err(error) = listen(callback) {
println!("Error: {:?}", error)
// Have this main thread recieve events and send them to the key handler thread
{
let callback = move |event: Event| {
tx.send(event).unwrap();
};

// This will block.
if let Err(error) = listen(callback) {
println!("Error: {:?}", error)
}
}

Ok(())
Expand Down

0 comments on commit 0414b18

Please sign in to comment.