use super::open_parent;
#[cfg(unix)]
use crate::fs::{append_dir_suffix, path_has_trailing_slash};
use crate::fs::{rename_unchecked, strip_dir_suffix, MaybeOwnedFile};
use std::path::Path;
use std::{fs, io};
/// Implement `rename` by `open`ing up the parent component of the path and
/// then calling `rename_unchecked` on the last component.
pub(crate) fn rename(
old_start: &fs::File,
old_path: &Path,
new_start: &fs::File,
new_path: &Path,
) -> io::Result<()> {
let old_start = MaybeOwnedFile::borrowed(old_start);
let new_start = MaybeOwnedFile::borrowed(new_start);
// As a special case, `rename` ignores a trailing slash rather than treating
// it as equivalent to a trailing slash-dot, so strip any trailing slashes
// for the purposes of `open_parent`.
//
// And on Unix, remember whether the source started with a slash so that we
// can still fail if it is and the source is a regular file.
#[cfg(unix)]
let old_starts_with_slash = path_has_trailing_slash(old_path);
let old_path = strip_dir_suffix(old_path);
let new_path = strip_dir_suffix(new_path);
let (old_dir, old_basename) = open_parent(old_start, &old_path)?;
let (new_dir, new_basename) = open_parent(new_start, &new_path)?;
// On Unix, re-append a slash if needed.
#[cfg(unix)]
let concat;
#[cfg(unix)]
let old_basename = if old_starts_with_slash {
concat = append_dir_suffix(old_basename.to_owned().into());
concat.as_os_str()
} else {
old_basename
};
rename_unchecked(
&old_dir,
old_basename.as_ref(),
&new_dir,
new_basename.as_ref(),
)
}