From f5072b8e0f798b891c0228b92a2a24edbac99b1b Mon Sep 17 00:00:00 2001 From: Konrad Borowski Date: Mon, 25 Mar 2019 08:02:33 +0100 Subject: [PATCH] Allow non-Unicode file names on Unix-like platforms --- src/lib.rs | 48 +++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 47 insertions(+), 1 deletion(-) diff --git a/src/lib.rs b/src/lib.rs index 3685349..98785d5 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -276,6 +276,13 @@ fn len_as_c_int(len: usize) -> Result { } } +#[cfg(unix)] +fn path_to_cstring(p: &Path) -> Result { + use std::os::unix::ffi::OsStrExt; + Ok(CString::new(p.as_os_str().as_bytes())?) +} + +#[cfg(not(unix))] fn path_to_cstring(p: &Path) -> Result { let s = p.to_str().ok_or_else(|| Error::InvalidPath(p.to_owned()))?; str_to_cstring(s) @@ -836,7 +843,17 @@ unsafe fn db_filename(db: *mut ffi::sqlite3) -> Option { if db_filename.is_null() { None } else { - CStr::from_ptr(db_filename).to_str().ok().map(PathBuf::from) + let cstr = CStr::from_ptr(db_filename); + #[cfg(unix)] + { + use std::ffi::OsStr; + use std::os::unix::ffi::OsStrExt; + Some(OsStr::from_bytes(cstr.to_bytes()).into()) + } + #[cfg(not(unix))] + { + cstr.to_str().ok().map(PathBuf::from) + } } } #[cfg(not(feature = "bundled"))] @@ -939,6 +956,35 @@ mod test { assert_eq!(42i64, the_answer.unwrap()); } + #[cfg(unix)] + #[test] + fn test_invalid_unicode_file_names() { + use std::ffi::OsStr; + use std::fs::File; + use std::os::unix::ffi::OsStrExt; + let temp_dir = TempDir::new("test_invalid_unicode").unwrap(); + let path = temp_dir.path(); + if File::create(path.join(OsStr::from_bytes(&[0xFE]))).is_err() { + // Skip test, filesystem doesn't support invalid Unicode + return; + } + let db_path = path.join(OsStr::from_bytes(&[0xFF])); + + { + let db = Connection::open(&db_path).unwrap(); + let sql = "BEGIN; + CREATE TABLE foo(x INTEGER); + INSERT INTO foo VALUES(42); + END;"; + db.execute_batch(sql).unwrap(); + } + + let db = Connection::open(&db_path).unwrap(); + let the_answer: Result = db.query_row("SELECT x FROM foo", NO_PARAMS, |r| r.get(0)); + + assert_eq!(42i64, the_answer.unwrap()); + } + #[test] fn test_open() { assert!(Connection::open_in_memory().is_ok());