TokioMiddleExperience
Какие ошибки делают команды, когда используют Tokio (Rust) без понимания его модели выполнения или сборки?
Самые частые ошибки: blocking в async контексте, неконтролируемый spawn без backpressure, игнорирование cancellation, неправильное использование Mutex и запуск CPU-bound кода в async runtime.
Типичные ошибки команд без понимания модели выполнения
Ошибка 1: Blocking в async контексте
// ПЛОХО: блокирует весь worker-поток
async fn handle_request() {
std::thread::sleep(Duration::from_secs(1)); // блокирует поток!
let data = std::fs::read_to_string("file.txt").unwrap(); // блокирует!
}
// ХОРОШО
async fn handle_request() {
tokio::time::sleep(Duration::from_secs(1)).await;
let data = tokio::fs::read_to_string("file.txt").await.unwrap();
// или для CPU-bound:
let result = tokio::task::spawn_blocking(|| {
expensive_computation()
}).await.unwrap();
}
Ошибка 2: Бесконтрольный spawn без backpressure
// ПЛОХО: queue растёт бесконтрольно при burst
while let Some(job) = rx.recv().await {
tokio::spawn(process(job)); // spawn без ограничения
}
// ХОРОШО: Semaphore ограничивает конкурентность
let sem = Arc::new(tokio::sync::Semaphore::new(100));
while let Some(job) = rx.recv().await {
let permit = sem.clone().acquire_owned().await.unwrap();
tokio::spawn(async move {
let _permit = permit; // держим до завершения задачи
process(job).await;
});
}
Ошибка 3: Игнорирование cancellation
// ПЛОХО: при cancel ресурс не освобождается
async fn write_to_db(conn: &mut Conn, data: Data) {
conn.begin_transaction().await.unwrap();
conn.insert(data).await.unwrap();
// если здесь task отменена, транзакция не закоммичена
// НО и не откатана — соединение в сломанном состоянии
conn.commit().await.unwrap();
}
// ХОРОШО: использовать RAII guard или явный rollback в Drop
struct Transaction<'a> { conn: &'a mut Conn, committed: bool }
impl Drop for Transaction<'_> {
fn drop(&mut self) {
if !self.committed {
// rollback в Drop — синхронно, без .await
}
}
}
Ошибка 4: tokio::sync::Mutex везде
// ПЛОХО: async Mutex дорогой, если critical section синхронная
let data = tokio::sync::Mutex::new(HashMap::new());
// ХОРОШО: для синхронных операций — std::sync::Mutex
// Правило: если внутри lock нет .await — используй std::sync::Mutex
let data = std::sync::Mutex::new(HashMap::new());
// НЕ держать std::sync::Mutex через .await — deadlock!
Ошибка 5: Забытые JoinHandle
// ПЛОХО: паника внутри spawn незаметна
tokio::spawn(async { panic!("oops"); });
// ХОРОШО: сохраняем handle и обрабатываем результат
let handle = tokio::spawn(async { panic!("oops"); });
match handle.await {
Ok(_) => {},
Err(e) if e.is_panic() => eprintln!("Task panicked!"),
Err(e) => eprintln!("Task cancelled: {e}"),
}
Ошибка 6: Неправильный выбор flavour runtime
// В тестах на CI создаём несколько #[tokio::test] — каждый создаёт свой runtime
// Если тест создаёт ещё один runtime внутри — паника
// Используйте Handle::try_current() для проверки
// Для Lambda/CLI — current_thread экономит потоки
#[tokio::main(flavor = "current_thread")]
async fn main() { /* ... */ }
Подводные камни
- rayon внутри tokio::spawn: Rayon блокирует при ожидании worker-потока — deadlock если tokio-потоки == rayon-потоки.
- tracing::subscriber не установлен: без
tracing_subscriber::fmt::init()все span-ы молча теряются. - Неправильный shutdown: без
runtime.shutdown_timeout()задачи могут не завершиться корректно при остановке сервиса. - select! без default ветки:
tokio::select!при отмене одной ветки дропает все остальные futures — можно потерять данные из канала. - broadcast::Receiver lag: если receiver медленный, он пропускает сообщения с ошибкой
RecvError::Lagged— часто игнорируется командой. - interval() первый tick немедленный:
tokio::time::interval()срабатывает сразу при создании. Если нужна задержка — используйтеinterval_at(Instant::now() + period, period). - async в trait не стабилизирован полностью: без
async-traitмакроса или nightly у вас boxing overhead на каждый вызов.
What hurts your answer
- Перечислять ошибки без объяснения причин
- Не отличать beginner mistakes от production failure modes
- Не предлагать процесс, который предотвращает повторение ошибок
What they're listening for
- Знает типичные ошибки при работе с Tokio (Rust)
- Понимает причины ошибок
- Предлагает практики prevention и early detection