1#![warn(clippy::all, clippy::pedantic, clippy::undocumented_unsafe_blocks)]
2#![allow(
3 clippy::let_underscore_untyped,
4 reason = "https://github.com/rust-lang/rust-clippy/pull/10442#issuecomment-1516570154"
5)]
6#![allow(
7 clippy::question_mark,
8 reason = "https://github.com/rust-lang/rust-clippy/issues/8281"
9)]
10#![allow(clippy::manual_let_else, reason = "manual_let_else was very buggy on release")]
11#![allow(clippy::missing_errors_doc, reason = "A lot of existing code fails this lint")]
12#![allow(
13 clippy::unnecessary_lazy_evaluations,
14 reason = "https://github.com/rust-lang/rust-clippy/issues/8109"
15)]
16#![cfg_attr(
17 test,
18 allow(clippy::non_ascii_literal, reason = "tests sometimes require UTF-8 string content")
19)]
20#![allow(unknown_lints)]
21#![warn(
22 missing_copy_implementations,
23 missing_debug_implementations,
24 missing_docs,
25 rust_2024_compatibility,
26 trivial_casts,
27 trivial_numeric_casts,
28 unused_qualifications,
29 variant_size_differences
30)]
31#![cfg_attr(docsrs, feature(doc_cfg))]
36#![cfg_attr(docsrs, feature(doc_alias))]
37
38#[cfg(doctest)]
69#[doc = include_str!("../README.md")]
70mod readme {}
71
72use std::path::PathBuf;
73
74#[must_use]
120pub fn repl_history_file() -> Option<PathBuf> {
121 let data_dir = repl_history_dir()?;
122
123 #[cfg(not(any(test, doctest, miri)))] let _ignored = std::fs::create_dir_all(&data_dir);
137
138 Some(data_dir.join(history_file_basename()))
139}
140
141#[must_use]
142#[cfg(target_vendor = "apple")]
143fn repl_history_dir() -> Option<PathBuf> {
144 use std::env;
145 use std::ffi::{CStr, OsString, c_char};
146 use std::os::unix::ffi::OsStringExt;
147
148 use sysdir::{
149 PATH_MAX, SYSDIR_DOMAIN_MASK_USER, sysdir_get_next_search_path_enumeration, sysdir_search_path_directory_t,
150 sysdir_start_search_path_enumeration,
151 };
152
153 let mut path = [0; PATH_MAX as usize];
168
169 let dir = sysdir_search_path_directory_t::SYSDIR_DIRECTORY_APPLICATION_SUPPORT;
170 let domain_mask = SYSDIR_DOMAIN_MASK_USER;
171
172 let application_support_bytes = unsafe {
178 let mut state = sysdir_start_search_path_enumeration(dir, domain_mask);
180 let path = path.as_mut_ptr().cast::<c_char>();
181 state = sysdir_get_next_search_path_enumeration(state, path);
182 if state.is_finished() {
183 return None;
184 }
185 let path = CStr::from_ptr(path);
186 path.to_bytes()
187 };
188
189 #[allow(deprecated)]
203 let application_support = match application_support_bytes {
204 [] => return None,
205 [b'~'] => env::home_dir()?,
206 [b'~', b'/', tail @ ..] => {
213 let home = env::home_dir()?;
214 let mut home = home.into_os_string().into_vec();
215
216 home.try_reserve_exact(1 + tail.len()).ok()?;
217 home.push(b'/');
218 home.extend_from_slice(tail);
219
220 OsString::from_vec(home).into()
221 }
222 path => {
223 let mut buf = vec![];
224 buf.try_reserve_exact(path.len()).ok()?;
225 buf.extend_from_slice(path);
226 OsString::from_vec(buf).into()
227 }
228 };
229 Some(application_support.join("org.artichokeruby.airb"))
233}
234
235#[must_use]
236#[cfg(all(unix, not(target_vendor = "apple")))]
237fn repl_history_dir() -> Option<PathBuf> {
238 use std::env;
239
240 let state_dir = match env::var_os("XDG_STATE_HOME") {
250 Some(path) if path.is_empty() => None,
252 Some(path) => Some(path),
253 None => None,
255 };
256
257 let state_dir = if let Some(state_dir) = state_dir {
258 PathBuf::from(state_dir)
259 } else {
260 #[allow(deprecated)]
274 let mut state_dir = env::home_dir()?;
275 state_dir.extend([".local", "state"]);
276 state_dir
277 };
278
279 Some(state_dir.join("artichokeruby"))
280}
281
282#[must_use]
283#[cfg(windows)]
284fn repl_history_dir() -> Option<PathBuf> {
285 use known_folders::{KnownFolder, get_known_folder_path};
286
287 let local_app_data = get_known_folder_path(KnownFolder::LocalAppData)?;
288 Some(local_app_data.join("Artichoke Ruby").join("airb").join("data"))
289}
290
291#[must_use]
300fn history_file_basename() -> &'static str {
301 if cfg!(windows) {
302 return "history.txt";
303 }
304 if cfg!(target_vendor = "apple") {
305 return "history";
306 }
307 if cfg!(unix) {
308 return "airb_history";
309 }
310 "history"
311}
312
313#[cfg(test)]
314mod tests {
315 use std::ffi::OsStr;
316 use std::path;
317
318 use super::*;
319
320 #[cfg(all(unix, not(target_vendor = "apple")))]
322 static ENV_LOCK: std::sync::Mutex<()> = std::sync::Mutex::new(());
323
324 #[test]
325 fn history_file_basename_is_non_empty() {
326 assert!(!history_file_basename().is_empty());
327 }
328
329 #[test]
330 fn history_file_basename_does_not_contain_path_separators() {
331 let filename = history_file_basename();
332 for c in filename.chars() {
333 assert!(!path::is_separator(c));
334 }
335 }
336
337 #[test]
338 fn history_file_basename_is_all_ascii() {
339 let filename = history_file_basename();
340 assert!(filename.is_ascii());
341 }
342
343 #[test]
344 fn history_file_basename_contains_the_word_history() {
345 let filename = history_file_basename();
346 assert!(filename.contains("history"));
347 }
348
349 #[test]
350 #[cfg(target_vendor = "apple")]
351 fn history_dir_on_apple_targets() {
352 let dir = repl_history_dir().unwrap();
353 let mut components = dir.components();
354
355 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("/"));
356 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Users"));
357 let _skip_user_dir = components.next().unwrap();
358 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Library"));
359 assert_eq!(
360 components.next().unwrap().as_os_str(),
361 OsStr::new("Application Support")
362 );
363 assert_eq!(
364 components.next().unwrap().as_os_str(),
365 OsStr::new("org.artichokeruby.airb")
366 );
367 assert!(components.next().is_none());
368 }
369
370 #[test]
371 #[cfg(target_vendor = "apple")]
372 fn history_file_on_apple_targets() {
373 let file = repl_history_file().unwrap();
374 let mut components = file.components();
375
376 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("/"));
377 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Users"));
378 let _skip_user_dir = components.next().unwrap();
379 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Library"));
380 assert_eq!(
381 components.next().unwrap().as_os_str(),
382 OsStr::new("Application Support")
383 );
384 assert_eq!(
385 components.next().unwrap().as_os_str(),
386 OsStr::new("org.artichokeruby.airb")
387 );
388 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("history"));
389 assert!(components.next().is_none());
390 }
391
392 #[test]
393 #[cfg(all(unix, not(target_vendor = "apple")))]
394 fn history_dir_on_unix_xdg_unset() {
395 use std::env;
396
397 let _guard = ENV_LOCK.lock();
398
399 unsafe {
401 env::remove_var("XDG_STATE_HOME");
402 }
403
404 let dir = repl_history_dir().unwrap();
405 let mut components = dir.components();
406
407 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("/"));
408 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("home"));
409 let _skip_user_dir = components.next().unwrap();
410 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new(".local"));
411 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("state"));
412 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("artichokeruby"));
413 assert!(components.next().is_none());
414 }
415
416 #[test]
417 #[cfg(all(unix, not(target_vendor = "apple")))]
418 fn history_file_on_unix_xdg_unset() {
419 use std::env;
420
421 let _guard = ENV_LOCK.lock();
422
423 unsafe {
425 env::remove_var("XDG_STATE_HOME");
426 }
427
428 let file = repl_history_file().unwrap();
429 let mut components = file.components();
430
431 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("/"));
432 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("home"));
433 let _skip_user_dir = components.next().unwrap();
434 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new(".local"));
435 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("state"));
436 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("artichokeruby"));
437 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("airb_history"));
438 assert!(components.next().is_none());
439 }
440
441 #[test]
442 #[cfg(all(unix, not(target_vendor = "apple")))]
443 fn history_dir_on_unix_empty_xdg_state_dir() {
444 use std::env;
445
446 let _guard = ENV_LOCK.lock();
447
448 unsafe {
450 env::remove_var("XDG_STATE_HOME");
451 env::set_var("XDG_STATE_HOME", "");
452 }
453
454 let dir = repl_history_dir().unwrap();
455 let mut components = dir.components();
456
457 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("/"));
458 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("home"));
459 let _skip_user_dir = components.next().unwrap();
460 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new(".local"));
461 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("state"));
462 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("artichokeruby"));
463 assert!(components.next().is_none());
464 }
465
466 #[test]
467 #[cfg(all(unix, not(target_vendor = "apple")))]
468 fn history_file_on_unix_empty_xdg_state_dir() {
469 use std::env;
470
471 let _guard = ENV_LOCK.lock();
472
473 unsafe {
475 env::remove_var("XDG_STATE_HOME");
476 env::set_var("XDG_STATE_HOME", "");
477 }
478
479 let file = repl_history_file().unwrap();
480 let mut components = file.components();
481
482 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("/"));
483 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("home"));
484 let _skip_user_dir = components.next().unwrap();
485 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new(".local"));
486 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("state"));
487 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("artichokeruby"));
488 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("airb_history"));
489 assert!(components.next().is_none());
490 }
491
492 #[test]
493 #[cfg(all(unix, not(target_vendor = "apple")))]
494 fn history_dir_on_unix_set_xdg_state_dir() {
495 use std::env;
496
497 let _guard = ENV_LOCK.lock();
498
499 unsafe {
501 env::remove_var("XDG_STATE_HOME");
502 env::set_var("XDG_STATE_HOME", "/opt/artichoke/state");
503 }
504
505 let dir = repl_history_dir().unwrap();
506 let mut components = dir.components();
507
508 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("/"));
509 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("opt"));
510 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("artichoke"));
511 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("state"));
512 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("artichokeruby"));
513 assert!(components.next().is_none());
514 }
515
516 #[test]
517 #[cfg(all(unix, not(target_vendor = "apple")))]
518 fn history_file_on_unix_set_xdg_state_dir() {
519 use std::env;
520
521 let _guard = ENV_LOCK.lock();
522
523 unsafe {
525 env::remove_var("XDG_STATE_HOME");
526 env::set_var("XDG_STATE_HOME", "/opt/artichoke/state");
527 }
528
529 let file = repl_history_file().unwrap();
530 let mut components = file.components();
531
532 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("/"));
533 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("opt"));
534 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("artichoke"));
535 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("state"));
536 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("artichokeruby"));
537 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("airb_history"));
538 assert!(components.next().is_none());
539 }
540
541 #[test]
542 #[cfg(windows)]
543 fn history_dir_on_windows() {
544 let dir = repl_history_dir().unwrap();
545 let mut components = dir.components();
546
547 let _skip_prefix = components.next().unwrap();
548 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new(r"\"));
549 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Users"));
550 let _skip_user_dir = components.next().unwrap();
551 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("AppData"));
552 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Local"));
553 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Artichoke Ruby"));
554 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("airb"));
555 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("data"));
556 assert!(components.next().is_none());
557 }
558
559 #[test]
560 #[cfg(windows)]
561 fn history_file_on_windows() {
562 let file = repl_history_file().unwrap();
563 let mut components = file.components();
564
565 let _skip_prefix = components.next().unwrap();
566 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new(r"\"));
567 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Users"));
568 let _skip_user_dir = components.next().unwrap();
569 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("AppData"));
570 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Local"));
571 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("Artichoke Ruby"));
572 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("airb"));
573 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("data"));
574 assert_eq!(components.next().unwrap().as_os_str(), OsStr::new("history.txt"));
575 assert!(components.next().is_none());
576 }
577}