Optimize full file status via passing in known file mtime
This commit is contained in:
parent
51d311affd
commit
6c09782aa2
2 changed files with 50 additions and 18 deletions
|
@ -1,6 +1,6 @@
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use collections::HashMap;
|
use collections::HashMap;
|
||||||
use git2::{BranchType, ErrorCode};
|
use git2::{BranchType, StatusShow};
|
||||||
use parking_lot::Mutex;
|
use parking_lot::Mutex;
|
||||||
use rpc::proto;
|
use rpc::proto;
|
||||||
use serde_derive::{Deserialize, Serialize};
|
use serde_derive::{Deserialize, Serialize};
|
||||||
|
@ -28,9 +28,25 @@ pub trait GitRepository: Send {
|
||||||
fn reload_index(&self);
|
fn reload_index(&self);
|
||||||
fn load_index_text(&self, relative_file_path: &Path) -> Option<String>;
|
fn load_index_text(&self, relative_file_path: &Path) -> Option<String>;
|
||||||
fn branch_name(&self) -> Option<String>;
|
fn branch_name(&self) -> Option<String>;
|
||||||
|
|
||||||
|
/// Get the statuses of all of the files in the index that start with the given
|
||||||
|
/// path and have changes with resepect to the HEAD commit. This is fast because
|
||||||
|
/// the index stores hashes of trees, so that unchanged directories can be skipped.
|
||||||
fn staged_statuses(&self, path_prefix: &Path) -> TreeMap<RepoPath, GitFileStatus>;
|
fn staged_statuses(&self, path_prefix: &Path) -> TreeMap<RepoPath, GitFileStatus>;
|
||||||
|
|
||||||
|
/// Get the status of a given file in the working directory with respect to
|
||||||
|
/// the index. In the common case, when there are no changes, this only requires
|
||||||
|
/// an index lookup. The index stores the mtime of each file when it was added,
|
||||||
|
/// so there's no work to do if the mtime matches.
|
||||||
fn unstaged_status(&self, path: &RepoPath, mtime: SystemTime) -> Option<GitFileStatus>;
|
fn unstaged_status(&self, path: &RepoPath, mtime: SystemTime) -> Option<GitFileStatus>;
|
||||||
fn status(&self, path: &RepoPath) -> Result<Option<GitFileStatus>>;
|
|
||||||
|
/// Get the status of a given file in the working directory with respect to
|
||||||
|
/// the HEAD commit. In the common case, when there are no changes, this only
|
||||||
|
/// requires an index lookup and blob comparison between the index and the HEAD
|
||||||
|
/// commit. The index stores the mtime of each file when it was added, so there's
|
||||||
|
/// no need to consider the working directory file if the mtime matches.
|
||||||
|
fn status(&self, path: &RepoPath, mtime: SystemTime) -> Option<GitFileStatus>;
|
||||||
|
|
||||||
fn branches(&self) -> Result<Vec<Branch>>;
|
fn branches(&self) -> Result<Vec<Branch>>;
|
||||||
fn change_branch(&self, _: &str) -> Result<()>;
|
fn change_branch(&self, _: &str) -> Result<()>;
|
||||||
fn create_branch(&self, _: &str) -> Result<()>;
|
fn create_branch(&self, _: &str) -> Result<()>;
|
||||||
|
@ -42,7 +58,6 @@ impl std::fmt::Debug for dyn GitRepository {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait::async_trait]
|
|
||||||
impl GitRepository for LibGitRepository {
|
impl GitRepository for LibGitRepository {
|
||||||
fn reload_index(&self) {
|
fn reload_index(&self) {
|
||||||
if let Ok(mut index) = self.index() {
|
if let Ok(mut index) = self.index() {
|
||||||
|
@ -122,18 +137,21 @@ impl GitRepository for LibGitRepository {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn status(&self, path: &RepoPath) -> Result<Option<GitFileStatus>> {
|
fn status(&self, path: &RepoPath, mtime: SystemTime) -> Option<GitFileStatus> {
|
||||||
let status = self.status_file(path);
|
let mut options = git2::StatusOptions::new();
|
||||||
match status {
|
options.pathspec(&path.0);
|
||||||
Ok(status) => Ok(read_status(status)),
|
options.disable_pathspec_match(true);
|
||||||
Err(e) => {
|
|
||||||
if e.code() == ErrorCode::NotFound {
|
// If the file has not changed since it was added to the index, then
|
||||||
Ok(None)
|
// there's no need to examine the working directory file: just compare
|
||||||
} else {
|
// the blob in the index to the one in the HEAD commit.
|
||||||
Err(e.into())
|
if matches_index(self, path, mtime) {
|
||||||
}
|
options.show(StatusShow::Index);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let statuses = self.statuses(Some(&mut options)).log_err()?;
|
||||||
|
let status = statuses.get(0).and_then(|s| read_status(s.status()));
|
||||||
|
status
|
||||||
}
|
}
|
||||||
|
|
||||||
fn branches(&self) -> Result<Vec<Branch>> {
|
fn branches(&self) -> Result<Vec<Branch>> {
|
||||||
|
@ -178,6 +196,21 @@ impl GitRepository for LibGitRepository {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn matches_index(repo: &LibGitRepository, path: &RepoPath, mtime: SystemTime) -> bool {
|
||||||
|
if let Some(index) = repo.index().log_err() {
|
||||||
|
if let Some(entry) = index.get_path(&path, 0) {
|
||||||
|
if let Some(mtime) = mtime.duration_since(SystemTime::UNIX_EPOCH).log_err() {
|
||||||
|
if entry.mtime.seconds() == mtime.as_secs() as i32
|
||||||
|
&& entry.mtime.nanoseconds() == mtime.subsec_nanos()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
false
|
||||||
|
}
|
||||||
|
|
||||||
fn read_status(status: git2::Status) -> Option<GitFileStatus> {
|
fn read_status(status: git2::Status) -> Option<GitFileStatus> {
|
||||||
if status.contains(git2::Status::CONFLICTED) {
|
if status.contains(git2::Status::CONFLICTED) {
|
||||||
Some(GitFileStatus::Conflict)
|
Some(GitFileStatus::Conflict)
|
||||||
|
@ -242,9 +275,9 @@ impl GitRepository for FakeGitRepository {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|
||||||
fn status(&self, path: &RepoPath) -> Result<Option<GitFileStatus>> {
|
fn status(&self, path: &RepoPath, _mtime: SystemTime) -> Option<GitFileStatus> {
|
||||||
let state = self.state.lock();
|
let state = self.state.lock();
|
||||||
Ok(state.worktree_statuses.get(path).cloned())
|
state.worktree_statuses.get(path).cloned()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn branches(&self) -> Result<Vec<Branch>> {
|
fn branches(&self) -> Result<Vec<Branch>> {
|
||||||
|
|
|
@ -3658,8 +3658,7 @@ impl BackgroundScanner {
|
||||||
if let Ok(repo_path) = path.strip_prefix(work_dir.0) {
|
if let Ok(repo_path) = path.strip_prefix(work_dir.0) {
|
||||||
let repo_path = RepoPath(repo_path.into());
|
let repo_path = RepoPath(repo_path.into());
|
||||||
let repo = repo.repo_ptr.lock();
|
let repo = repo.repo_ptr.lock();
|
||||||
fs_entry.git_status =
|
fs_entry.git_status = repo.status(&repo_path, fs_entry.mtime);
|
||||||
repo.status(&repo_path).log_err().flatten();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue