mirror of
https://github.com/zed-industries/zed.git
synced 2026-06-01 03:14:56 +07:00
Sometimes the action log would not auto-accept deleted hunks, and I
finally found a repro for this, here's an explanation of what went wrong
```rust
// Before
use crate::{Alpha, Beta};
fn keep() {
work();
}
fn remove() {
work();
}
fn after() {
work();
}
```
```rust
// After commit
use crate::{Alpha};
fn keep() {
work();
}
fn after() {
work();
}
```
The action log may track the deletion as:
```diff
fn keep() {
work();
}
-fn remove() {
- work();
-}
-
fn after() {
work();
}
```
But the commit diff may choose different boundaries because nearby lines
repeat:
```diff
fn keep() {
work();
-}
-
-fn remove() {
- work();
}
fn after() {
work();
}
```
Both diffs produce the same final file, but their row ranges differ. The
previous logic only accepted committed edits when those row ranges
matched exactly, so already-committed edits could remain marked as
unaccepted.
Now we have a fast path for checking if the base_text matches exactly,
which works fine in this case.
Release Notes:
- Fixed an issue where agent edits would sometimes not get auto-accepted
when they were commited
3520 lines
127 KiB
Rust
3520 lines
127 KiB
Rust
use anyhow::{Context as _, Result};
|
|
use buffer_diff::BufferDiff;
|
|
use clock;
|
|
use collections::{BTreeMap, HashMap};
|
|
use fs::MTime;
|
|
use futures::{FutureExt, StreamExt, channel::mpsc};
|
|
use gpui::{
|
|
App, AppContext, AsyncApp, Context, Entity, SharedString, Subscription, Task, WeakEntity,
|
|
};
|
|
use language::{Anchor, Buffer, BufferEvent, Point, ToOffset, ToPoint};
|
|
use project::{Project, ProjectItem, lsp_store::OpenLspBufferHandle};
|
|
use std::{
|
|
cmp,
|
|
ops::Range,
|
|
path::{Path, PathBuf},
|
|
sync::Arc,
|
|
};
|
|
use text::{Edit, Patch, Rope};
|
|
use util::{RangeExt, ResultExt as _};
|
|
|
|
/// Stores undo information for a single buffer's rejected edits
|
|
#[derive(Clone)]
|
|
pub struct PerBufferUndo {
|
|
pub buffer: WeakEntity<Buffer>,
|
|
pub edits_to_restore: Vec<(Range<Anchor>, String)>,
|
|
pub status: UndoBufferStatus,
|
|
}
|
|
|
|
/// Tracks the buffer status for undo purposes
|
|
#[derive(Clone, Debug)]
|
|
pub enum UndoBufferStatus {
|
|
Modified,
|
|
/// Buffer was created by the agent.
|
|
/// - `had_existing_content: true` - Agent overwrote an existing file. On reject, the
|
|
/// original content was restored. Undo is supported: we restore the agent's content.
|
|
/// - `had_existing_content: false` - Agent created a new file that didn't exist before.
|
|
/// On reject, the file was deleted. Undo is NOT currently supported (would require
|
|
/// recreating the file). Future TODO.
|
|
Created {
|
|
had_existing_content: bool,
|
|
},
|
|
}
|
|
|
|
/// Stores undo information for the most recent reject operation
|
|
#[derive(Clone)]
|
|
pub struct LastRejectUndo {
|
|
/// Per-buffer undo information
|
|
pub buffers: Vec<PerBufferUndo>,
|
|
}
|
|
|
|
/// Tracks actions performed by tools in a thread
|
|
pub struct ActionLog {
|
|
/// Buffers that we want to notify the model about when they change.
|
|
tracked_buffers: BTreeMap<Entity<Buffer>, TrackedBuffer>,
|
|
/// The project this action log is associated with
|
|
project: Entity<Project>,
|
|
/// An action log to forward all public methods to
|
|
/// Useful in cases like subagents, where we want to track individual diffs for this subagent,
|
|
/// but also want to associate the reads/writes with a parent review experience
|
|
linked_action_log: Option<Entity<ActionLog>>,
|
|
/// Stores undo information for the most recent reject operation
|
|
last_reject_undo: Option<LastRejectUndo>,
|
|
/// Tracks the last time files were read by the agent, to detect external modifications
|
|
file_read_times: HashMap<PathBuf, MTime>,
|
|
}
|
|
|
|
impl ActionLog {
|
|
/// Creates a new, empty action log associated with the given project.
|
|
pub fn new(project: Entity<Project>) -> Self {
|
|
Self {
|
|
tracked_buffers: BTreeMap::default(),
|
|
project,
|
|
linked_action_log: None,
|
|
last_reject_undo: None,
|
|
file_read_times: HashMap::default(),
|
|
}
|
|
}
|
|
|
|
pub fn with_linked_action_log(mut self, linked_action_log: Entity<ActionLog>) -> Self {
|
|
self.linked_action_log = Some(linked_action_log);
|
|
self
|
|
}
|
|
|
|
pub fn project(&self) -> &Entity<Project> {
|
|
&self.project
|
|
}
|
|
|
|
pub fn file_read_time(&self, path: &Path) -> Option<MTime> {
|
|
self.file_read_times.get(path).copied()
|
|
}
|
|
|
|
fn update_file_read_time(&mut self, buffer: &Entity<Buffer>, cx: &App) {
|
|
let buffer = buffer.read(cx);
|
|
if let Some(file) = buffer.file() {
|
|
if let Some(local_file) = file.as_local() {
|
|
if let Some(mtime) = file.disk_state().mtime() {
|
|
let abs_path = local_file.abs_path(cx);
|
|
self.file_read_times.insert(abs_path, mtime);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn remove_file_read_time(&mut self, buffer: &Entity<Buffer>, cx: &App) {
|
|
let buffer = buffer.read(cx);
|
|
if let Some(file) = buffer.file() {
|
|
if let Some(local_file) = file.as_local() {
|
|
let abs_path = local_file.abs_path(cx);
|
|
self.file_read_times.remove(&abs_path);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn track_buffer_internal(
|
|
&mut self,
|
|
buffer: Entity<Buffer>,
|
|
is_created: bool,
|
|
cx: &mut Context<Self>,
|
|
) -> &mut TrackedBuffer {
|
|
let status = if is_created {
|
|
if let Some(tracked) = self.tracked_buffers.remove(&buffer) {
|
|
match tracked.status {
|
|
TrackedBufferStatus::Created {
|
|
existing_file_content,
|
|
} => TrackedBufferStatus::Created {
|
|
existing_file_content,
|
|
},
|
|
TrackedBufferStatus::Modified | TrackedBufferStatus::Deleted => {
|
|
TrackedBufferStatus::Created {
|
|
existing_file_content: Some(tracked.diff_base),
|
|
}
|
|
}
|
|
}
|
|
} else if buffer
|
|
.read(cx)
|
|
.file()
|
|
.is_some_and(|file| file.disk_state().exists())
|
|
{
|
|
TrackedBufferStatus::Created {
|
|
existing_file_content: Some(buffer.read(cx).as_rope().clone()),
|
|
}
|
|
} else {
|
|
TrackedBufferStatus::Created {
|
|
existing_file_content: None,
|
|
}
|
|
}
|
|
} else {
|
|
TrackedBufferStatus::Modified
|
|
};
|
|
|
|
let tracked_buffer = self
|
|
.tracked_buffers
|
|
.entry(buffer.clone())
|
|
.or_insert_with(|| {
|
|
let open_lsp_handle = self.project.update(cx, |project, cx| {
|
|
project.register_buffer_with_language_servers(&buffer, cx)
|
|
});
|
|
|
|
let text_snapshot = buffer.read(cx).text_snapshot();
|
|
let language = buffer.read(cx).language().cloned();
|
|
let language_registry = buffer.read(cx).language_registry();
|
|
let diff = cx.new(|cx| {
|
|
let mut diff = BufferDiff::new(&text_snapshot, cx);
|
|
diff.language_changed(language, language_registry, cx);
|
|
diff
|
|
});
|
|
let (diff_update_tx, diff_update_rx) = mpsc::unbounded();
|
|
let diff_base;
|
|
let unreviewed_edits;
|
|
if is_created {
|
|
diff_base = Rope::default();
|
|
unreviewed_edits = Patch::new(vec![Edit {
|
|
old: 0..1,
|
|
new: 0..text_snapshot.max_point().row + 1,
|
|
}])
|
|
} else {
|
|
diff_base = buffer.read(cx).as_rope().clone();
|
|
unreviewed_edits = Patch::default();
|
|
}
|
|
TrackedBuffer {
|
|
buffer: buffer.clone(),
|
|
diff_base,
|
|
unreviewed_edits,
|
|
snapshot: text_snapshot,
|
|
status,
|
|
version: buffer.read(cx).version(),
|
|
diff,
|
|
diff_update: diff_update_tx,
|
|
_open_lsp_handle: open_lsp_handle,
|
|
_maintain_diff: cx.spawn({
|
|
let buffer = buffer.clone();
|
|
async move |this, cx| {
|
|
Self::maintain_diff(this, buffer, diff_update_rx, cx)
|
|
.await
|
|
.ok();
|
|
}
|
|
}),
|
|
_subscription: cx.subscribe(&buffer, Self::handle_buffer_event),
|
|
}
|
|
});
|
|
tracked_buffer.version = buffer.read(cx).version();
|
|
tracked_buffer
|
|
}
|
|
|
|
fn handle_buffer_event(
|
|
&mut self,
|
|
buffer: Entity<Buffer>,
|
|
event: &BufferEvent,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
match event {
|
|
BufferEvent::Edited { .. } => {
|
|
let Some(tracked_buffer) = self.tracked_buffers.get_mut(&buffer) else {
|
|
return;
|
|
};
|
|
let buffer_version = buffer.read(cx).version();
|
|
if !buffer_version.changed_since(&tracked_buffer.version) {
|
|
return;
|
|
}
|
|
self.handle_buffer_edited(buffer, cx);
|
|
}
|
|
BufferEvent::FileHandleChanged => {
|
|
self.handle_buffer_file_changed(buffer, cx);
|
|
}
|
|
_ => {}
|
|
};
|
|
}
|
|
|
|
fn handle_buffer_edited(&mut self, buffer: Entity<Buffer>, cx: &mut Context<Self>) {
|
|
let Some(tracked_buffer) = self.tracked_buffers.get_mut(&buffer) else {
|
|
return;
|
|
};
|
|
tracked_buffer.schedule_diff_update(ChangeAuthor::User, cx);
|
|
}
|
|
|
|
fn handle_buffer_file_changed(&mut self, buffer: Entity<Buffer>, cx: &mut Context<Self>) {
|
|
let Some(tracked_buffer) = self.tracked_buffers.get_mut(&buffer) else {
|
|
return;
|
|
};
|
|
|
|
match tracked_buffer.status {
|
|
TrackedBufferStatus::Created { .. } | TrackedBufferStatus::Modified => {
|
|
if buffer
|
|
.read(cx)
|
|
.file()
|
|
.is_some_and(|file| file.disk_state().is_deleted())
|
|
{
|
|
// If the buffer had been edited by a tool, but it got
|
|
// deleted externally, we want to stop tracking it.
|
|
self.tracked_buffers.remove(&buffer);
|
|
}
|
|
cx.notify();
|
|
}
|
|
TrackedBufferStatus::Deleted => {
|
|
if buffer
|
|
.read(cx)
|
|
.file()
|
|
.is_some_and(|file| !file.disk_state().is_deleted())
|
|
{
|
|
// If the buffer had been deleted by a tool, but it got
|
|
// resurrected externally, we want to clear the edits we
|
|
// were tracking and reset the buffer's state.
|
|
self.tracked_buffers.remove(&buffer);
|
|
self.track_buffer_internal(buffer, false, cx);
|
|
}
|
|
cx.notify();
|
|
}
|
|
}
|
|
}
|
|
|
|
async fn maintain_diff(
|
|
this: WeakEntity<Self>,
|
|
buffer: Entity<Buffer>,
|
|
mut buffer_updates: mpsc::UnboundedReceiver<(ChangeAuthor, text::BufferSnapshot)>,
|
|
cx: &mut AsyncApp,
|
|
) -> Result<()> {
|
|
let git_diff = this
|
|
.update(cx, |this, cx| {
|
|
this.project.update(cx, |project, cx| {
|
|
project.open_uncommitted_diff(buffer.clone(), cx)
|
|
})
|
|
})?
|
|
.await
|
|
.ok();
|
|
let (mut git_diff_updates_tx, mut git_diff_updates_rx) = watch::channel(());
|
|
let _diff_subscription = if let Some(git_diff) = git_diff.as_ref() {
|
|
cx.update(|cx| {
|
|
Some(cx.subscribe(git_diff, move |_, event, _cx| {
|
|
if matches!(event, buffer_diff::BufferDiffEvent::BaseTextChanged) {
|
|
git_diff_updates_tx.send(()).ok();
|
|
}
|
|
}))
|
|
})
|
|
} else {
|
|
None
|
|
};
|
|
|
|
loop {
|
|
futures::select_biased! {
|
|
buffer_update = buffer_updates.next() => {
|
|
if let Some((author, buffer_snapshot)) = buffer_update {
|
|
Self::track_edits(&this, &buffer, author, buffer_snapshot, cx).await?;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
_ = git_diff_updates_rx.changed().fuse() => {
|
|
if let Some(git_diff) = git_diff.as_ref() {
|
|
Self::keep_committed_edits(&this, &buffer, git_diff, cx).await?;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn track_edits(
|
|
this: &WeakEntity<ActionLog>,
|
|
buffer: &Entity<Buffer>,
|
|
author: ChangeAuthor,
|
|
buffer_snapshot: text::BufferSnapshot,
|
|
cx: &mut AsyncApp,
|
|
) -> Result<()> {
|
|
let rebase = this.update(cx, |this, cx| {
|
|
let tracked_buffer = this
|
|
.tracked_buffers
|
|
.get_mut(buffer)
|
|
.context("buffer not tracked")?;
|
|
|
|
let rebase = cx.background_spawn({
|
|
let mut base_text = tracked_buffer.diff_base.clone();
|
|
let old_snapshot = tracked_buffer.snapshot.clone();
|
|
let new_snapshot = buffer_snapshot.clone();
|
|
let unreviewed_edits = tracked_buffer.unreviewed_edits.clone();
|
|
let edits = diff_snapshots(&old_snapshot, &new_snapshot);
|
|
async move {
|
|
if let ChangeAuthor::User = author {
|
|
apply_non_conflicting_edits(
|
|
&unreviewed_edits,
|
|
edits,
|
|
&mut base_text,
|
|
new_snapshot.as_rope(),
|
|
);
|
|
}
|
|
|
|
(Arc::from(base_text.to_string().as_str()), base_text)
|
|
}
|
|
});
|
|
|
|
anyhow::Ok(rebase)
|
|
})??;
|
|
let (new_base_text, new_diff_base) = rebase.await;
|
|
|
|
Self::update_diff(
|
|
this,
|
|
buffer,
|
|
buffer_snapshot,
|
|
new_base_text,
|
|
new_diff_base,
|
|
cx,
|
|
)
|
|
.await
|
|
}
|
|
|
|
async fn keep_committed_edits(
|
|
this: &WeakEntity<ActionLog>,
|
|
buffer: &Entity<Buffer>,
|
|
git_diff: &Entity<BufferDiff>,
|
|
cx: &mut AsyncApp,
|
|
) -> Result<()> {
|
|
let buffer_snapshot = this.read_with(cx, |this, _cx| {
|
|
let tracked_buffer = this
|
|
.tracked_buffers
|
|
.get(buffer)
|
|
.context("buffer not tracked")?;
|
|
anyhow::Ok(tracked_buffer.snapshot.clone())
|
|
})??;
|
|
let (new_base_text, new_diff_base) = this
|
|
.read_with(cx, |this, cx| {
|
|
let tracked_buffer = this
|
|
.tracked_buffers
|
|
.get(buffer)
|
|
.context("buffer not tracked")?;
|
|
let old_unreviewed_edits = tracked_buffer.unreviewed_edits.clone();
|
|
let agent_diff_base = tracked_buffer.diff_base.clone();
|
|
let git_diff_base = git_diff.read(cx).base_text(cx).as_rope().clone();
|
|
let buffer_text = tracked_buffer.snapshot.as_rope().clone();
|
|
anyhow::Ok(cx.background_spawn(async move {
|
|
if buffer_text.len() == git_diff_base.len()
|
|
&& buffer_text.chars_at(0).eq(git_diff_base.chars_at(0))
|
|
{
|
|
return (Arc::<str>::from(git_diff_base.to_string()), git_diff_base);
|
|
}
|
|
let mut old_unreviewed_edits = old_unreviewed_edits.into_iter().peekable();
|
|
let committed_edits = language::line_diff(
|
|
&agent_diff_base.to_string(),
|
|
&git_diff_base.to_string(),
|
|
)
|
|
.into_iter()
|
|
.map(|(old, new)| Edit { old, new });
|
|
|
|
let mut new_agent_diff_base = agent_diff_base.clone();
|
|
let mut row_delta = 0i32;
|
|
for committed in committed_edits {
|
|
while let Some(unreviewed) = old_unreviewed_edits.peek() {
|
|
// If the committed edit matches the unreviewed
|
|
// edit, assume the user wants to keep it.
|
|
if committed.old == unreviewed.old {
|
|
let unreviewed_new =
|
|
buffer_text.slice_rows(unreviewed.new.clone()).to_string();
|
|
let committed_new =
|
|
git_diff_base.slice_rows(committed.new.clone()).to_string();
|
|
if unreviewed_new == committed_new {
|
|
let old_byte_start =
|
|
new_agent_diff_base.point_to_offset(Point::new(
|
|
(unreviewed.old.start as i32 + row_delta) as u32,
|
|
0,
|
|
));
|
|
let old_byte_end =
|
|
new_agent_diff_base.point_to_offset(cmp::min(
|
|
Point::new(
|
|
(unreviewed.old.end as i32 + row_delta) as u32,
|
|
0,
|
|
),
|
|
new_agent_diff_base.max_point(),
|
|
));
|
|
new_agent_diff_base
|
|
.replace(old_byte_start..old_byte_end, &unreviewed_new);
|
|
row_delta +=
|
|
unreviewed.new_len() as i32 - unreviewed.old_len() as i32;
|
|
}
|
|
} else if unreviewed.old.start >= committed.old.end {
|
|
break;
|
|
}
|
|
|
|
old_unreviewed_edits.next().unwrap();
|
|
}
|
|
}
|
|
|
|
(
|
|
Arc::from(new_agent_diff_base.to_string().as_str()),
|
|
new_agent_diff_base,
|
|
)
|
|
}))
|
|
})??
|
|
.await;
|
|
|
|
Self::update_diff(
|
|
this,
|
|
buffer,
|
|
buffer_snapshot,
|
|
new_base_text,
|
|
new_diff_base,
|
|
cx,
|
|
)
|
|
.await
|
|
}
|
|
|
|
async fn update_diff(
|
|
this: &WeakEntity<ActionLog>,
|
|
buffer: &Entity<Buffer>,
|
|
buffer_snapshot: text::BufferSnapshot,
|
|
new_base_text: Arc<str>,
|
|
new_diff_base: Rope,
|
|
cx: &mut AsyncApp,
|
|
) -> Result<()> {
|
|
let (diff, language) = this.read_with(cx, |this, cx| {
|
|
let tracked_buffer = this
|
|
.tracked_buffers
|
|
.get(buffer)
|
|
.context("buffer not tracked")?;
|
|
anyhow::Ok((
|
|
tracked_buffer.diff.clone(),
|
|
buffer.read(cx).language().cloned(),
|
|
))
|
|
})??;
|
|
let update = diff
|
|
.update(cx, |diff, cx| {
|
|
diff.update_diff(
|
|
buffer_snapshot.clone(),
|
|
Some(new_base_text),
|
|
Some(true),
|
|
language,
|
|
cx,
|
|
)
|
|
})
|
|
.await;
|
|
diff.update(cx, |diff, cx| {
|
|
diff.set_snapshot(update.clone(), &buffer_snapshot, cx)
|
|
})
|
|
.await;
|
|
let diff_snapshot = diff.update(cx, |diff, cx| diff.snapshot(cx));
|
|
|
|
let unreviewed_edits = cx
|
|
.background_spawn({
|
|
let buffer_snapshot = buffer_snapshot.clone();
|
|
let new_diff_base = new_diff_base.clone();
|
|
async move {
|
|
let mut unreviewed_edits = Patch::default();
|
|
for hunk in diff_snapshot.hunks_intersecting_range(
|
|
Anchor::min_for_buffer(buffer_snapshot.remote_id())
|
|
..Anchor::max_for_buffer(buffer_snapshot.remote_id()),
|
|
&buffer_snapshot,
|
|
) {
|
|
let old_range = new_diff_base
|
|
.offset_to_point(hunk.diff_base_byte_range.start)
|
|
..new_diff_base.offset_to_point(hunk.diff_base_byte_range.end);
|
|
let new_range = hunk.range.start..hunk.range.end;
|
|
unreviewed_edits.push(point_to_row_edit(
|
|
Edit {
|
|
old: old_range,
|
|
new: new_range,
|
|
},
|
|
&new_diff_base,
|
|
buffer_snapshot.as_rope(),
|
|
));
|
|
}
|
|
unreviewed_edits
|
|
}
|
|
})
|
|
.await;
|
|
this.update(cx, |this, cx| {
|
|
let tracked_buffer = this
|
|
.tracked_buffers
|
|
.get_mut(buffer)
|
|
.context("buffer not tracked")?;
|
|
tracked_buffer.diff_base = new_diff_base;
|
|
tracked_buffer.snapshot = buffer_snapshot;
|
|
tracked_buffer.unreviewed_edits = unreviewed_edits;
|
|
cx.notify();
|
|
anyhow::Ok(())
|
|
})?
|
|
}
|
|
|
|
/// Track a buffer as read by agent, so we can notify the model about user edits.
|
|
pub fn buffer_read(&mut self, buffer: Entity<Buffer>, cx: &mut Context<Self>) {
|
|
self.buffer_read_impl(buffer, true, cx);
|
|
}
|
|
|
|
fn buffer_read_impl(
|
|
&mut self,
|
|
buffer: Entity<Buffer>,
|
|
record_file_read_time: bool,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
if let Some(linked_action_log) = &self.linked_action_log {
|
|
// We don't want to share read times since the other agent hasn't read it necessarily
|
|
linked_action_log.update(cx, |log, cx| {
|
|
log.buffer_read_impl(buffer.clone(), false, cx);
|
|
});
|
|
}
|
|
if record_file_read_time {
|
|
self.update_file_read_time(&buffer, cx);
|
|
}
|
|
self.track_buffer_internal(buffer, false, cx);
|
|
}
|
|
|
|
/// Mark a buffer as created by agent, so we can refresh it in the context
|
|
pub fn buffer_created(&mut self, buffer: Entity<Buffer>, cx: &mut Context<Self>) {
|
|
self.buffer_created_impl(buffer, true, cx);
|
|
}
|
|
|
|
fn buffer_created_impl(
|
|
&mut self,
|
|
buffer: Entity<Buffer>,
|
|
record_file_read_time: bool,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
if let Some(linked_action_log) = &self.linked_action_log {
|
|
// We don't want to share read times since the other agent hasn't read it necessarily
|
|
linked_action_log.update(cx, |log, cx| {
|
|
log.buffer_created_impl(buffer.clone(), false, cx);
|
|
});
|
|
}
|
|
if record_file_read_time {
|
|
self.update_file_read_time(&buffer, cx);
|
|
}
|
|
self.track_buffer_internal(buffer, true, cx);
|
|
}
|
|
|
|
/// Mark a buffer as edited by agent, so we can refresh it in the context
|
|
pub fn buffer_edited(&mut self, buffer: Entity<Buffer>, cx: &mut Context<Self>) {
|
|
self.buffer_edited_impl(buffer, true, cx);
|
|
}
|
|
|
|
fn buffer_edited_impl(
|
|
&mut self,
|
|
buffer: Entity<Buffer>,
|
|
record_file_read_time: bool,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
if let Some(linked_action_log) = &self.linked_action_log {
|
|
// We don't want to share read times since the other agent hasn't read it necessarily
|
|
linked_action_log.update(cx, |log, cx| {
|
|
log.buffer_edited_impl(buffer.clone(), false, cx);
|
|
});
|
|
}
|
|
if record_file_read_time {
|
|
self.update_file_read_time(&buffer, cx);
|
|
}
|
|
let new_version = buffer.read(cx).version();
|
|
let tracked_buffer = self.track_buffer_internal(buffer, false, cx);
|
|
if let TrackedBufferStatus::Deleted = tracked_buffer.status {
|
|
tracked_buffer.status = TrackedBufferStatus::Modified;
|
|
}
|
|
|
|
tracked_buffer.version = new_version;
|
|
tracked_buffer.schedule_diff_update(ChangeAuthor::Agent, cx);
|
|
}
|
|
|
|
pub fn will_delete_buffer(&mut self, buffer: Entity<Buffer>, cx: &mut Context<Self>) {
|
|
// Ok to propagate file read time removal to linked action log
|
|
self.remove_file_read_time(&buffer, cx);
|
|
let has_linked_action_log = self.linked_action_log.is_some();
|
|
let tracked_buffer = self.track_buffer_internal(buffer.clone(), false, cx);
|
|
match tracked_buffer.status {
|
|
TrackedBufferStatus::Created { .. } => {
|
|
self.tracked_buffers.remove(&buffer);
|
|
cx.notify();
|
|
}
|
|
TrackedBufferStatus::Modified => {
|
|
tracked_buffer.status = TrackedBufferStatus::Deleted;
|
|
if !has_linked_action_log {
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("", cx));
|
|
tracked_buffer.schedule_diff_update(ChangeAuthor::Agent, cx);
|
|
}
|
|
}
|
|
|
|
TrackedBufferStatus::Deleted => {}
|
|
}
|
|
|
|
if let Some(linked_action_log) = &mut self.linked_action_log {
|
|
linked_action_log.update(cx, |log, cx| log.will_delete_buffer(buffer.clone(), cx));
|
|
}
|
|
|
|
if has_linked_action_log && let Some(tracked_buffer) = self.tracked_buffers.get(&buffer) {
|
|
tracked_buffer.schedule_diff_update(ChangeAuthor::Agent, cx);
|
|
}
|
|
|
|
cx.notify();
|
|
}
|
|
|
|
pub fn keep_edits_in_range(
|
|
&mut self,
|
|
buffer: Entity<Buffer>,
|
|
buffer_range: Range<impl language::ToPoint>,
|
|
telemetry: Option<ActionLogTelemetry>,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
let Some(tracked_buffer) = self.tracked_buffers.get_mut(&buffer) else {
|
|
return;
|
|
};
|
|
|
|
let mut metrics = ActionLogMetrics::for_buffer(buffer.read(cx));
|
|
match tracked_buffer.status {
|
|
TrackedBufferStatus::Deleted => {
|
|
metrics.add_edits(tracked_buffer.unreviewed_edits.edits());
|
|
self.tracked_buffers.remove(&buffer);
|
|
cx.notify();
|
|
}
|
|
_ => {
|
|
let buffer = buffer.read(cx);
|
|
let buffer_range =
|
|
buffer_range.start.to_point(buffer)..buffer_range.end.to_point(buffer);
|
|
let mut delta = 0i32;
|
|
tracked_buffer.unreviewed_edits.retain_mut(|edit| {
|
|
edit.old.start = (edit.old.start as i32 + delta) as u32;
|
|
edit.old.end = (edit.old.end as i32 + delta) as u32;
|
|
|
|
if buffer_range.end.row < edit.new.start
|
|
|| buffer_range.start.row > edit.new.end
|
|
{
|
|
true
|
|
} else {
|
|
let old_range = tracked_buffer
|
|
.diff_base
|
|
.point_to_offset(Point::new(edit.old.start, 0))
|
|
..tracked_buffer.diff_base.point_to_offset(cmp::min(
|
|
Point::new(edit.old.end, 0),
|
|
tracked_buffer.diff_base.max_point(),
|
|
));
|
|
let new_range = tracked_buffer
|
|
.snapshot
|
|
.point_to_offset(Point::new(edit.new.start, 0))
|
|
..tracked_buffer.snapshot.point_to_offset(cmp::min(
|
|
Point::new(edit.new.end, 0),
|
|
tracked_buffer.snapshot.max_point(),
|
|
));
|
|
tracked_buffer.diff_base.replace(
|
|
old_range,
|
|
&tracked_buffer
|
|
.snapshot
|
|
.text_for_range(new_range)
|
|
.collect::<String>(),
|
|
);
|
|
delta += edit.new_len() as i32 - edit.old_len() as i32;
|
|
metrics.add_edit(edit);
|
|
false
|
|
}
|
|
});
|
|
if tracked_buffer.unreviewed_edits.is_empty()
|
|
&& let TrackedBufferStatus::Created { .. } = &mut tracked_buffer.status
|
|
{
|
|
tracked_buffer.status = TrackedBufferStatus::Modified;
|
|
}
|
|
tracked_buffer.schedule_diff_update(ChangeAuthor::User, cx);
|
|
}
|
|
}
|
|
if let Some(telemetry) = telemetry {
|
|
telemetry_report_accepted_edits(&telemetry, metrics);
|
|
}
|
|
}
|
|
|
|
pub fn reject_edits_in_ranges(
|
|
&mut self,
|
|
buffer: Entity<Buffer>,
|
|
buffer_ranges: Vec<Range<impl language::ToPoint>>,
|
|
telemetry: Option<ActionLogTelemetry>,
|
|
cx: &mut Context<Self>,
|
|
) -> (Task<Result<()>>, Option<PerBufferUndo>) {
|
|
let Some(tracked_buffer) = self.tracked_buffers.get_mut(&buffer) else {
|
|
return (Task::ready(Ok(())), None);
|
|
};
|
|
|
|
let mut metrics = ActionLogMetrics::for_buffer(buffer.read(cx));
|
|
let mut undo_info: Option<PerBufferUndo> = None;
|
|
let task = match &tracked_buffer.status {
|
|
TrackedBufferStatus::Created {
|
|
existing_file_content,
|
|
} => {
|
|
let task = if let Some(existing_file_content) = existing_file_content {
|
|
// Capture the agent's content before restoring existing file content
|
|
let agent_content = buffer.read(cx).text();
|
|
let buffer_id = buffer.read(cx).remote_id();
|
|
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.start_transaction();
|
|
buffer.set_text("", cx);
|
|
for chunk in existing_file_content.chunks() {
|
|
buffer.append(chunk, cx);
|
|
}
|
|
buffer.end_transaction(cx);
|
|
});
|
|
|
|
undo_info = Some(PerBufferUndo {
|
|
buffer: buffer.downgrade(),
|
|
edits_to_restore: vec![(
|
|
Anchor::min_for_buffer(buffer_id)..Anchor::max_for_buffer(buffer_id),
|
|
agent_content,
|
|
)],
|
|
status: UndoBufferStatus::Created {
|
|
had_existing_content: true,
|
|
},
|
|
});
|
|
|
|
self.project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
} else {
|
|
// For a file created by AI with no pre-existing content,
|
|
// only delete the file if we're certain it contains only AI content
|
|
// with no edits from the user.
|
|
|
|
let initial_version = tracked_buffer.version.clone();
|
|
let current_version = buffer.read(cx).version();
|
|
|
|
let current_content = buffer.read(cx).text();
|
|
let tracked_content = tracked_buffer.snapshot.text();
|
|
|
|
let is_ai_only_content =
|
|
initial_version == current_version && current_content == tracked_content;
|
|
|
|
if is_ai_only_content {
|
|
let task = buffer
|
|
.read(cx)
|
|
.entry_id(cx)
|
|
.and_then(|entry_id| {
|
|
self.project.update(cx, |project, cx| {
|
|
project.delete_entry(entry_id, false, cx)
|
|
})
|
|
})
|
|
.unwrap_or_else(|| Task::ready(Ok(None)));
|
|
|
|
cx.background_spawn(async move {
|
|
task.await?;
|
|
Ok(())
|
|
})
|
|
} else {
|
|
// Not sure how to disentangle edits made by the user
|
|
// from edits made by the AI at this point.
|
|
// For now, preserve both to avoid data loss.
|
|
//
|
|
// TODO: Better solution (disable "Reject" after user makes some
|
|
// edit or find a way to differentiate between AI and user edits)
|
|
Task::ready(Ok(()))
|
|
}
|
|
};
|
|
|
|
metrics.add_edits(tracked_buffer.unreviewed_edits.edits());
|
|
self.tracked_buffers.remove(&buffer);
|
|
cx.notify();
|
|
task
|
|
}
|
|
TrackedBufferStatus::Deleted => {
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.set_text(tracked_buffer.diff_base.to_string(), cx)
|
|
});
|
|
let save = self
|
|
.project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx));
|
|
|
|
// Clear all tracked edits for this buffer and start over as if we just read it.
|
|
metrics.add_edits(tracked_buffer.unreviewed_edits.edits());
|
|
self.tracked_buffers.remove(&buffer);
|
|
self.buffer_read(buffer.clone(), cx);
|
|
cx.notify();
|
|
save
|
|
}
|
|
TrackedBufferStatus::Modified => {
|
|
let edits_to_restore = buffer.update(cx, |buffer, cx| {
|
|
let mut buffer_row_ranges = buffer_ranges
|
|
.into_iter()
|
|
.map(|range| {
|
|
range.start.to_point(buffer).row..range.end.to_point(buffer).row
|
|
})
|
|
.peekable();
|
|
|
|
let mut edits_to_revert = Vec::new();
|
|
let mut edits_for_undo = Vec::new();
|
|
for edit in tracked_buffer.unreviewed_edits.edits() {
|
|
let new_range = tracked_buffer
|
|
.snapshot
|
|
.anchor_before(Point::new(edit.new.start, 0))
|
|
..tracked_buffer.snapshot.anchor_after(cmp::min(
|
|
Point::new(edit.new.end, 0),
|
|
tracked_buffer.snapshot.max_point(),
|
|
));
|
|
let new_row_range = new_range.start.to_point(buffer).row
|
|
..new_range.end.to_point(buffer).row;
|
|
|
|
let mut revert = false;
|
|
while let Some(buffer_row_range) = buffer_row_ranges.peek() {
|
|
if buffer_row_range.end < new_row_range.start {
|
|
buffer_row_ranges.next();
|
|
} else if buffer_row_range.start > new_row_range.end {
|
|
break;
|
|
} else {
|
|
revert = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if revert {
|
|
metrics.add_edit(edit);
|
|
let old_range = tracked_buffer
|
|
.diff_base
|
|
.point_to_offset(Point::new(edit.old.start, 0))
|
|
..tracked_buffer.diff_base.point_to_offset(cmp::min(
|
|
Point::new(edit.old.end, 0),
|
|
tracked_buffer.diff_base.max_point(),
|
|
));
|
|
let old_text = tracked_buffer
|
|
.diff_base
|
|
.chunks_in_range(old_range)
|
|
.collect::<String>();
|
|
|
|
// Capture the agent's text before we revert it (for undo)
|
|
let new_range_offset =
|
|
new_range.start.to_offset(buffer)..new_range.end.to_offset(buffer);
|
|
let agent_text =
|
|
buffer.text_for_range(new_range_offset).collect::<String>();
|
|
edits_for_undo.push((new_range.clone(), agent_text));
|
|
|
|
edits_to_revert.push((new_range, old_text));
|
|
}
|
|
}
|
|
|
|
buffer.edit(edits_to_revert, None, cx);
|
|
edits_for_undo
|
|
});
|
|
|
|
if !edits_to_restore.is_empty() {
|
|
undo_info = Some(PerBufferUndo {
|
|
buffer: buffer.downgrade(),
|
|
edits_to_restore,
|
|
status: UndoBufferStatus::Modified,
|
|
});
|
|
}
|
|
|
|
self.project
|
|
.update(cx, |project, cx| project.save_buffer(buffer, cx))
|
|
}
|
|
};
|
|
if let Some(telemetry) = telemetry {
|
|
telemetry_report_rejected_edits(&telemetry, metrics);
|
|
}
|
|
(task, undo_info)
|
|
}
|
|
|
|
pub fn keep_all_edits(
|
|
&mut self,
|
|
telemetry: Option<ActionLogTelemetry>,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
self.tracked_buffers.retain(|buffer, tracked_buffer| {
|
|
let mut metrics = ActionLogMetrics::for_buffer(buffer.read(cx));
|
|
metrics.add_edits(tracked_buffer.unreviewed_edits.edits());
|
|
if let Some(telemetry) = telemetry.as_ref() {
|
|
telemetry_report_accepted_edits(telemetry, metrics);
|
|
}
|
|
match tracked_buffer.status {
|
|
TrackedBufferStatus::Deleted => false,
|
|
_ => {
|
|
if let TrackedBufferStatus::Created { .. } = &mut tracked_buffer.status {
|
|
tracked_buffer.status = TrackedBufferStatus::Modified;
|
|
}
|
|
tracked_buffer.unreviewed_edits.clear();
|
|
tracked_buffer.diff_base = tracked_buffer.snapshot.as_rope().clone();
|
|
tracked_buffer.schedule_diff_update(ChangeAuthor::User, cx);
|
|
true
|
|
}
|
|
}
|
|
});
|
|
|
|
cx.notify();
|
|
}
|
|
|
|
pub fn reject_all_edits(
|
|
&mut self,
|
|
telemetry: Option<ActionLogTelemetry>,
|
|
cx: &mut Context<Self>,
|
|
) -> Task<()> {
|
|
// Clear any previous undo state before starting a new reject operation
|
|
self.last_reject_undo = None;
|
|
|
|
let mut undo_buffers = Vec::new();
|
|
let mut futures = Vec::new();
|
|
|
|
for buffer in self.changed_buffers(cx).into_keys() {
|
|
let buffer_ranges = vec![Anchor::min_max_range_for_buffer(
|
|
buffer.read(cx).remote_id(),
|
|
)];
|
|
let (reject_task, undo_info) =
|
|
self.reject_edits_in_ranges(buffer, buffer_ranges, telemetry.clone(), cx);
|
|
|
|
if let Some(undo) = undo_info {
|
|
undo_buffers.push(undo);
|
|
}
|
|
|
|
futures.push(async move {
|
|
reject_task.await.log_err();
|
|
});
|
|
}
|
|
|
|
// Store the undo information if we have any
|
|
if !undo_buffers.is_empty() {
|
|
self.last_reject_undo = Some(LastRejectUndo {
|
|
buffers: undo_buffers,
|
|
});
|
|
}
|
|
|
|
let task = futures::future::join_all(futures);
|
|
cx.background_spawn(async move {
|
|
task.await;
|
|
})
|
|
}
|
|
|
|
pub fn has_pending_undo(&self) -> bool {
|
|
self.last_reject_undo.is_some()
|
|
}
|
|
|
|
pub fn set_last_reject_undo(&mut self, undo: LastRejectUndo) {
|
|
self.last_reject_undo = Some(undo);
|
|
}
|
|
|
|
/// Undoes the most recent reject operation, restoring the rejected agent changes.
|
|
/// This is a best-effort operation: if buffers have been closed or modified externally,
|
|
/// those buffers will be skipped.
|
|
pub fn undo_last_reject(&mut self, cx: &mut Context<Self>) -> Task<()> {
|
|
let Some(undo) = self.last_reject_undo.take() else {
|
|
return Task::ready(());
|
|
};
|
|
|
|
let mut save_tasks = Vec::with_capacity(undo.buffers.len());
|
|
|
|
for per_buffer_undo in undo.buffers {
|
|
// Skip if the buffer entity has been deallocated
|
|
let Some(buffer) = per_buffer_undo.buffer.upgrade() else {
|
|
continue;
|
|
};
|
|
|
|
buffer.update(cx, |buffer, cx| {
|
|
let mut valid_edits = Vec::new();
|
|
|
|
for (anchor_range, text_to_restore) in per_buffer_undo.edits_to_restore {
|
|
if anchor_range.start.buffer_id == buffer.remote_id()
|
|
&& anchor_range.end.buffer_id == buffer.remote_id()
|
|
{
|
|
valid_edits.push((anchor_range, text_to_restore));
|
|
}
|
|
}
|
|
|
|
if !valid_edits.is_empty() {
|
|
buffer.edit(valid_edits, None, cx);
|
|
}
|
|
});
|
|
|
|
if !self.tracked_buffers.contains_key(&buffer) {
|
|
self.buffer_edited(buffer.clone(), cx);
|
|
}
|
|
|
|
let save = self
|
|
.project
|
|
.update(cx, |project, cx| project.save_buffer(buffer, cx));
|
|
save_tasks.push(save);
|
|
}
|
|
|
|
cx.notify();
|
|
|
|
cx.background_spawn(async move {
|
|
futures::future::join_all(save_tasks).await;
|
|
})
|
|
}
|
|
|
|
/// Returns the set of buffers that contain edits that haven't been reviewed by the user.
|
|
pub fn changed_buffers(&self, cx: &App) -> BTreeMap<Entity<Buffer>, Entity<BufferDiff>> {
|
|
self.tracked_buffers
|
|
.iter()
|
|
.filter(|(_, tracked)| tracked.has_edits(cx))
|
|
.map(|(buffer, tracked)| (buffer.clone(), tracked.diff.clone()))
|
|
.collect()
|
|
}
|
|
|
|
/// Returns the total number of lines added and removed across all unreviewed buffers.
|
|
pub fn diff_stats(&self, cx: &App) -> DiffStats {
|
|
DiffStats::all_files(&self.changed_buffers(cx), cx)
|
|
}
|
|
|
|
/// Iterate over buffers changed since last read or edited by the model
|
|
pub fn stale_buffers<'a>(&'a self, cx: &'a App) -> impl Iterator<Item = &'a Entity<Buffer>> {
|
|
self.tracked_buffers
|
|
.iter()
|
|
.filter(|(buffer, tracked)| {
|
|
let buffer = buffer.read(cx);
|
|
|
|
tracked.version != buffer.version
|
|
&& buffer
|
|
.file()
|
|
.is_some_and(|file| !file.disk_state().is_deleted())
|
|
})
|
|
.map(|(buffer, _)| buffer)
|
|
}
|
|
}
|
|
|
|
#[derive(Default, Debug, Clone, Copy)]
|
|
pub struct DiffStats {
|
|
pub lines_added: u32,
|
|
pub lines_removed: u32,
|
|
}
|
|
|
|
impl DiffStats {
|
|
pub fn single_file(buffer: &Buffer, diff: &BufferDiff, cx: &App) -> Self {
|
|
let mut stats = DiffStats::default();
|
|
let diff_snapshot = diff.snapshot(cx);
|
|
let buffer_snapshot = buffer.snapshot();
|
|
let base_text = diff_snapshot.base_text();
|
|
|
|
for hunk in diff_snapshot.hunks(&buffer_snapshot) {
|
|
let added_rows = hunk.range.end.row.saturating_sub(hunk.range.start.row);
|
|
stats.lines_added += added_rows;
|
|
|
|
let base_start = hunk.diff_base_byte_range.start.to_point(base_text).row;
|
|
let base_end = hunk.diff_base_byte_range.end.to_point(base_text).row;
|
|
let removed_rows = base_end.saturating_sub(base_start);
|
|
stats.lines_removed += removed_rows;
|
|
}
|
|
|
|
stats
|
|
}
|
|
|
|
pub fn all_files(
|
|
changed_buffers: &BTreeMap<Entity<Buffer>, Entity<BufferDiff>>,
|
|
cx: &App,
|
|
) -> Self {
|
|
let mut total = DiffStats::default();
|
|
for (buffer, diff) in changed_buffers {
|
|
let stats = DiffStats::single_file(buffer.read(cx), diff.read(cx), cx);
|
|
total.lines_added += stats.lines_added;
|
|
total.lines_removed += stats.lines_removed;
|
|
}
|
|
total
|
|
}
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub struct ActionLogTelemetry {
|
|
pub agent_telemetry_id: SharedString,
|
|
pub session_id: Arc<str>,
|
|
}
|
|
|
|
struct ActionLogMetrics {
|
|
lines_removed: u32,
|
|
lines_added: u32,
|
|
language: Option<SharedString>,
|
|
}
|
|
|
|
impl ActionLogMetrics {
|
|
fn for_buffer(buffer: &Buffer) -> Self {
|
|
Self {
|
|
language: buffer.language().map(|l| l.name().0),
|
|
lines_removed: 0,
|
|
lines_added: 0,
|
|
}
|
|
}
|
|
|
|
fn add_edits(&mut self, edits: &[Edit<u32>]) {
|
|
for edit in edits {
|
|
self.add_edit(edit);
|
|
}
|
|
}
|
|
|
|
fn add_edit(&mut self, edit: &Edit<u32>) {
|
|
self.lines_added += edit.new_len();
|
|
self.lines_removed += edit.old_len();
|
|
}
|
|
}
|
|
|
|
fn telemetry_report_accepted_edits(telemetry: &ActionLogTelemetry, metrics: ActionLogMetrics) {
|
|
telemetry::event!(
|
|
"Agent Edits Accepted",
|
|
agent = telemetry.agent_telemetry_id,
|
|
session = telemetry.session_id,
|
|
language = metrics.language,
|
|
lines_added = metrics.lines_added,
|
|
lines_removed = metrics.lines_removed
|
|
);
|
|
}
|
|
|
|
fn telemetry_report_rejected_edits(telemetry: &ActionLogTelemetry, metrics: ActionLogMetrics) {
|
|
telemetry::event!(
|
|
"Agent Edits Rejected",
|
|
agent = telemetry.agent_telemetry_id,
|
|
session = telemetry.session_id,
|
|
language = metrics.language,
|
|
lines_added = metrics.lines_added,
|
|
lines_removed = metrics.lines_removed
|
|
);
|
|
}
|
|
|
|
fn apply_non_conflicting_edits(
|
|
patch: &Patch<u32>,
|
|
edits: Vec<Edit<u32>>,
|
|
old_text: &mut Rope,
|
|
new_text: &Rope,
|
|
) -> bool {
|
|
let mut old_edits = patch.edits().iter().cloned().peekable();
|
|
let mut new_edits = edits.into_iter().peekable();
|
|
let mut applied_delta = 0i32;
|
|
let mut rebased_delta = 0i32;
|
|
let mut has_made_changes = false;
|
|
|
|
while let Some(mut new_edit) = new_edits.next() {
|
|
let mut conflict = false;
|
|
|
|
// Push all the old edits that are before this new edit or that intersect with it.
|
|
while let Some(old_edit) = old_edits.peek() {
|
|
if new_edit.old.end < old_edit.new.start
|
|
|| (!old_edit.new.is_empty() && new_edit.old.end == old_edit.new.start)
|
|
{
|
|
break;
|
|
} else if new_edit.old.start > old_edit.new.end
|
|
|| (!old_edit.new.is_empty() && new_edit.old.start == old_edit.new.end)
|
|
{
|
|
let old_edit = old_edits.next().unwrap();
|
|
rebased_delta += old_edit.new_len() as i32 - old_edit.old_len() as i32;
|
|
} else {
|
|
conflict = true;
|
|
if new_edits
|
|
.peek()
|
|
.is_some_and(|next_edit| next_edit.old.overlaps(&old_edit.new))
|
|
{
|
|
new_edit = new_edits.next().unwrap();
|
|
} else {
|
|
let old_edit = old_edits.next().unwrap();
|
|
rebased_delta += old_edit.new_len() as i32 - old_edit.old_len() as i32;
|
|
}
|
|
}
|
|
}
|
|
|
|
if !conflict {
|
|
// This edit doesn't intersect with any old edit, so we can apply it to the old text.
|
|
new_edit.old.start = (new_edit.old.start as i32 + applied_delta - rebased_delta) as u32;
|
|
new_edit.old.end = (new_edit.old.end as i32 + applied_delta - rebased_delta) as u32;
|
|
let old_bytes = old_text.point_to_offset(Point::new(new_edit.old.start, 0))
|
|
..old_text.point_to_offset(cmp::min(
|
|
Point::new(new_edit.old.end, 0),
|
|
old_text.max_point(),
|
|
));
|
|
let new_bytes = new_text.point_to_offset(Point::new(new_edit.new.start, 0))
|
|
..new_text.point_to_offset(cmp::min(
|
|
Point::new(new_edit.new.end, 0),
|
|
new_text.max_point(),
|
|
));
|
|
|
|
old_text.replace(
|
|
old_bytes,
|
|
&new_text.chunks_in_range(new_bytes).collect::<String>(),
|
|
);
|
|
applied_delta += new_edit.new_len() as i32 - new_edit.old_len() as i32;
|
|
has_made_changes = true;
|
|
}
|
|
}
|
|
has_made_changes
|
|
}
|
|
|
|
fn diff_snapshots(
|
|
old_snapshot: &text::BufferSnapshot,
|
|
new_snapshot: &text::BufferSnapshot,
|
|
) -> Vec<Edit<u32>> {
|
|
let mut edits = new_snapshot
|
|
.edits_since::<Point>(&old_snapshot.version)
|
|
.map(|edit| point_to_row_edit(edit, old_snapshot.as_rope(), new_snapshot.as_rope()))
|
|
.peekable();
|
|
let mut row_edits = Vec::new();
|
|
while let Some(mut edit) = edits.next() {
|
|
while let Some(next_edit) = edits.peek() {
|
|
if edit.old.end >= next_edit.old.start {
|
|
edit.old.end = next_edit.old.end;
|
|
edit.new.end = next_edit.new.end;
|
|
edits.next();
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
row_edits.push(edit);
|
|
}
|
|
row_edits
|
|
}
|
|
|
|
fn point_to_row_edit(edit: Edit<Point>, old_text: &Rope, new_text: &Rope) -> Edit<u32> {
|
|
if edit.old.start.column == old_text.line_len(edit.old.start.row)
|
|
&& new_text
|
|
.chars_at(new_text.point_to_offset(edit.new.start))
|
|
.next()
|
|
== Some('\n')
|
|
&& edit.old.start != old_text.max_point()
|
|
{
|
|
Edit {
|
|
old: edit.old.start.row + 1..edit.old.end.row + 1,
|
|
new: edit.new.start.row + 1..edit.new.end.row + 1,
|
|
}
|
|
} else if edit.old.start.column == 0 && edit.old.end.column == 0 && edit.new.end.column == 0 {
|
|
Edit {
|
|
old: edit.old.start.row..edit.old.end.row,
|
|
new: edit.new.start.row..edit.new.end.row,
|
|
}
|
|
} else {
|
|
Edit {
|
|
old: edit.old.start.row..edit.old.end.row + 1,
|
|
new: edit.new.start.row..edit.new.end.row + 1,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
enum ChangeAuthor {
|
|
User,
|
|
Agent,
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
enum TrackedBufferStatus {
|
|
Created { existing_file_content: Option<Rope> },
|
|
Modified,
|
|
Deleted,
|
|
}
|
|
|
|
pub struct TrackedBuffer {
|
|
buffer: Entity<Buffer>,
|
|
diff_base: Rope,
|
|
unreviewed_edits: Patch<u32>,
|
|
status: TrackedBufferStatus,
|
|
version: clock::Global,
|
|
diff: Entity<BufferDiff>,
|
|
snapshot: text::BufferSnapshot,
|
|
diff_update: mpsc::UnboundedSender<(ChangeAuthor, text::BufferSnapshot)>,
|
|
_open_lsp_handle: OpenLspBufferHandle,
|
|
_maintain_diff: Task<()>,
|
|
_subscription: Subscription,
|
|
}
|
|
|
|
impl TrackedBuffer {
|
|
#[cfg(any(test, feature = "test-support"))]
|
|
pub fn diff(&self) -> &Entity<BufferDiff> {
|
|
&self.diff
|
|
}
|
|
|
|
#[cfg(any(test, feature = "test-support"))]
|
|
pub fn diff_base_len(&self) -> usize {
|
|
self.diff_base.len()
|
|
}
|
|
|
|
fn has_edits(&self, cx: &App) -> bool {
|
|
self.diff
|
|
.read(cx)
|
|
.snapshot(cx)
|
|
.hunks(self.buffer.read(cx))
|
|
.next()
|
|
.is_some()
|
|
}
|
|
|
|
fn schedule_diff_update(&self, author: ChangeAuthor, cx: &App) {
|
|
self.diff_update
|
|
.unbounded_send((author, self.buffer.read(cx).text_snapshot()))
|
|
.ok();
|
|
}
|
|
}
|
|
|
|
pub struct ChangedBuffer {
|
|
pub diff: Entity<BufferDiff>,
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use buffer_diff::DiffHunkStatusKind;
|
|
use gpui::TestAppContext;
|
|
use indoc::indoc;
|
|
use language::Point;
|
|
use project::{FakeFs, Fs, Project, RemoveOptions};
|
|
use rand::prelude::*;
|
|
use serde_json::json;
|
|
use settings::SettingsStore;
|
|
use std::env;
|
|
use util::{RandomCharIter, path};
|
|
|
|
#[ctor::ctor]
|
|
fn init_logger() {
|
|
zlog::init_test();
|
|
}
|
|
|
|
fn init_test(cx: &mut TestAppContext) {
|
|
cx.update(|cx| {
|
|
let settings_store = SettingsStore::test(cx);
|
|
cx.set_global(settings_store);
|
|
});
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_keep_edits(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "abc\ndef\nghi\njkl\nmno"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(1, 1)..Point::new(1, 2), "E")], None, cx)
|
|
.unwrap()
|
|
});
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(4, 2)..Point::new(4, 3), "O")], None, cx)
|
|
.unwrap()
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndEf\nghi\njkl\nmnO"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![
|
|
HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(2, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "def\n".into(),
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(4, 0)..Point::new(4, 3),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "mno".into(),
|
|
}
|
|
],
|
|
)]
|
|
);
|
|
|
|
action_log.update(cx, |log, cx| {
|
|
log.keep_edits_in_range(buffer.clone(), Point::new(3, 0)..Point::new(4, 3), None, cx)
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(2, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "def\n".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
action_log.update(cx, |log, cx| {
|
|
log.keep_edits_in_range(buffer.clone(), Point::new(0, 0)..Point::new(4, 3), None, cx)
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_deletions(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(
|
|
path!("/dir"),
|
|
json!({"file": "abc\ndef\nghi\njkl\nmno\npqr"}),
|
|
)
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(1, 0)..Point::new(2, 0), "")], None, cx)
|
|
.unwrap();
|
|
buffer.finalize_last_transaction();
|
|
});
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(3, 0)..Point::new(4, 0), "")], None, cx)
|
|
.unwrap();
|
|
buffer.finalize_last_transaction();
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\nghi\njkl\npqr"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![
|
|
HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(1, 0),
|
|
diff_status: DiffHunkStatusKind::Deleted,
|
|
old_text: "def\n".into(),
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(3, 0)..Point::new(3, 0),
|
|
diff_status: DiffHunkStatusKind::Deleted,
|
|
old_text: "mno\n".into(),
|
|
}
|
|
],
|
|
)]
|
|
);
|
|
|
|
buffer.update(cx, |buffer, cx| buffer.undo(cx));
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\nghi\njkl\nmno\npqr"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(1, 0),
|
|
diff_status: DiffHunkStatusKind::Deleted,
|
|
old_text: "def\n".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
action_log.update(cx, |log, cx| {
|
|
log.keep_edits_in_range(buffer.clone(), Point::new(1, 0)..Point::new(1, 0), None, cx)
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_overlapping_user_edits(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "abc\ndef\nghi\njkl\nmno"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(1, 2)..Point::new(2, 3), "F\nGHI")], None, cx)
|
|
.unwrap()
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndeF\nGHI\njkl\nmno"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(3, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "def\nghi\n".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.edit(
|
|
[
|
|
(Point::new(0, 2)..Point::new(0, 2), "X"),
|
|
(Point::new(3, 0)..Point::new(3, 0), "Y"),
|
|
],
|
|
None,
|
|
cx,
|
|
)
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abXc\ndeF\nGHI\nYjkl\nmno"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(3, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "def\nghi\n".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.edit([(Point::new(1, 1)..Point::new(1, 1), "Z")], None, cx)
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abXc\ndZeF\nGHI\nYjkl\nmno"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(3, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "def\nghi\n".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
action_log.update(cx, |log, cx| {
|
|
log.keep_edits_in_range(buffer.clone(), Point::new(0, 0)..Point::new(1, 0), None, cx)
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_creating_files(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({})).await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file1", cx))
|
|
.unwrap();
|
|
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("lorem", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 5),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
buffer.update(cx, |buffer, cx| buffer.edit([(0..0, "X")], None, cx));
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 6),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
action_log.update(cx, |log, cx| {
|
|
log.keep_edits_in_range(buffer.clone(), 0..5, None, cx)
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_overwriting_files(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(
|
|
path!("/dir"),
|
|
json!({
|
|
"file1": "Lorem ipsum dolor"
|
|
}),
|
|
)
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file1", cx))
|
|
.unwrap();
|
|
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("sit amet consecteur", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 19),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let (task, _) = log.reject_edits_in_ranges(buffer.clone(), vec![2..5], None, cx);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _cx| buffer.text()),
|
|
"Lorem ipsum dolor"
|
|
);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_overwriting_previously_edited_files(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(
|
|
path!("/dir"),
|
|
json!({
|
|
"file1": "Lorem ipsum dolor"
|
|
}),
|
|
)
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file1", cx))
|
|
.unwrap();
|
|
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| buffer.append(" sit amet consecteur", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 37),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "Lorem ipsum dolor".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("rewritten", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 9),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let (task, _) = log.reject_edits_in_ranges(buffer.clone(), vec![2..5], None, cx);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _cx| buffer.text()),
|
|
"Lorem ipsum dolor"
|
|
);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_deleting_files(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(
|
|
path!("/dir"),
|
|
json!({"file1": "lorem\n", "file2": "ipsum\n"}),
|
|
)
|
|
.await;
|
|
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let file1_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file1", cx))
|
|
.unwrap();
|
|
let file2_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file2", cx))
|
|
.unwrap();
|
|
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let buffer1 = project
|
|
.update(cx, |project, cx| {
|
|
project.open_buffer(file1_path.clone(), cx)
|
|
})
|
|
.await
|
|
.unwrap();
|
|
let buffer2 = project
|
|
.update(cx, |project, cx| {
|
|
project.open_buffer(file2_path.clone(), cx)
|
|
})
|
|
.await
|
|
.unwrap();
|
|
|
|
action_log.update(cx, |log, cx| log.will_delete_buffer(buffer1.clone(), cx));
|
|
action_log.update(cx, |log, cx| log.will_delete_buffer(buffer2.clone(), cx));
|
|
project
|
|
.update(cx, |project, cx| {
|
|
project.delete_file(file1_path.clone(), false, cx)
|
|
})
|
|
.unwrap()
|
|
.await
|
|
.unwrap();
|
|
project
|
|
.update(cx, |project, cx| {
|
|
project.delete_file(file2_path.clone(), false, cx)
|
|
})
|
|
.unwrap()
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![
|
|
(
|
|
buffer1.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 0),
|
|
diff_status: DiffHunkStatusKind::Deleted,
|
|
old_text: "lorem\n".into(),
|
|
}]
|
|
),
|
|
(
|
|
buffer2.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 0),
|
|
diff_status: DiffHunkStatusKind::Deleted,
|
|
old_text: "ipsum\n".into(),
|
|
}],
|
|
)
|
|
]
|
|
);
|
|
|
|
// Simulate file1 being recreated externally.
|
|
fs.insert_file(path!("/dir/file1"), "LOREM".as_bytes().to_vec())
|
|
.await;
|
|
|
|
// Simulate file2 being recreated by a tool.
|
|
let buffer2 = project
|
|
.update(cx, |project, cx| project.open_buffer(file2_path, cx))
|
|
.await
|
|
.unwrap();
|
|
action_log.update(cx, |log, cx| log.buffer_created(buffer2.clone(), cx));
|
|
buffer2.update(cx, |buffer, cx| buffer.set_text("IPSUM", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer2.clone(), cx));
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer2.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer2.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 5),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
// Simulate file2 being deleted externally.
|
|
fs.remove_file(path!("/dir/file2").as_ref(), RemoveOptions::default())
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_reject_edits(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "abc\ndef\nghi\njkl\nmno"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(1, 1)..Point::new(1, 2), "E\nXYZ")], None, cx)
|
|
.unwrap()
|
|
});
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(5, 2)..Point::new(5, 3), "O")], None, cx)
|
|
.unwrap()
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndE\nXYZf\nghi\njkl\nmnO"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![
|
|
HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(3, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "def\n".into(),
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(5, 0)..Point::new(5, 3),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "mno".into(),
|
|
}
|
|
],
|
|
)]
|
|
);
|
|
|
|
// If the rejected range doesn't overlap with any hunk, we ignore it.
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let (task, _) = log.reject_edits_in_ranges(
|
|
buffer.clone(),
|
|
vec![Point::new(4, 0)..Point::new(4, 0)],
|
|
None,
|
|
cx,
|
|
);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndE\nXYZf\nghi\njkl\nmnO"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![
|
|
HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(3, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "def\n".into(),
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(5, 0)..Point::new(5, 3),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "mno".into(),
|
|
}
|
|
],
|
|
)]
|
|
);
|
|
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let (task, _) = log.reject_edits_in_ranges(
|
|
buffer.clone(),
|
|
vec![Point::new(0, 0)..Point::new(1, 0)],
|
|
None,
|
|
cx,
|
|
);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndef\nghi\njkl\nmnO"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(4, 0)..Point::new(4, 3),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "mno".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let (task, _) = log.reject_edits_in_ranges(
|
|
buffer.clone(),
|
|
vec![Point::new(4, 0)..Point::new(4, 0)],
|
|
None,
|
|
cx,
|
|
);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndef\nghi\njkl\nmno"
|
|
);
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_reject_multiple_edits(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "abc\ndef\nghi\njkl\nmno"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(1, 1)..Point::new(1, 2), "E\nXYZ")], None, cx)
|
|
.unwrap()
|
|
});
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(5, 2)..Point::new(5, 3), "O")], None, cx)
|
|
.unwrap()
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndE\nXYZf\nghi\njkl\nmnO"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![
|
|
HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(3, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "def\n".into(),
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(5, 0)..Point::new(5, 3),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "mno".into(),
|
|
}
|
|
],
|
|
)]
|
|
);
|
|
|
|
action_log.update(cx, |log, cx| {
|
|
let range_1 = buffer.read(cx).anchor_before(Point::new(0, 0))
|
|
..buffer.read(cx).anchor_before(Point::new(1, 0));
|
|
let range_2 = buffer.read(cx).anchor_before(Point::new(5, 0))
|
|
..buffer.read(cx).anchor_before(Point::new(5, 3));
|
|
|
|
let (task, _) =
|
|
log.reject_edits_in_ranges(buffer.clone(), vec![range_1, range_2], None, cx);
|
|
task.detach();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndef\nghi\njkl\nmno"
|
|
);
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndef\nghi\njkl\nmno"
|
|
);
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_reject_deleted_file(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "content"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.will_delete_buffer(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| {
|
|
project.delete_file(file_path.clone(), false, cx)
|
|
})
|
|
.unwrap()
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert!(!fs.is_file(path!("/dir/file").as_ref()).await);
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 0),
|
|
diff_status: DiffHunkStatusKind::Deleted,
|
|
old_text: "content".into(),
|
|
}]
|
|
)]
|
|
);
|
|
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let (task, _) = log.reject_edits_in_ranges(
|
|
buffer.clone(),
|
|
vec![Point::new(0, 0)..Point::new(0, 0)],
|
|
None,
|
|
cx,
|
|
);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_eq!(buffer.read_with(cx, |buffer, _| buffer.text()), "content");
|
|
assert!(fs.is_file(path!("/dir/file").as_ref()).await);
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test(iterations = 10)]
|
|
async fn test_reject_created_file(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path("dir/new_file", cx)
|
|
})
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("content", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
assert!(fs.is_file(path!("/dir/new_file").as_ref()).await);
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 7),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into(),
|
|
}],
|
|
)]
|
|
);
|
|
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let (task, _) = log.reject_edits_in_ranges(
|
|
buffer.clone(),
|
|
vec![Point::new(0, 0)..Point::new(0, 11)],
|
|
None,
|
|
cx,
|
|
);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert!(!fs.is_file(path!("/dir/new_file").as_ref()).await);
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_reject_created_file_with_user_edits(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path("dir/new_file", cx)
|
|
})
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
// AI creates file with initial content
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("ai content", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.run_until_parked();
|
|
|
|
// User makes additional edits
|
|
cx.update(|cx| {
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.edit([(10..10, "\nuser added this line")], None, cx);
|
|
});
|
|
});
|
|
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
assert!(fs.is_file(path!("/dir/new_file").as_ref()).await);
|
|
|
|
// Reject all
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let (task, _) = log.reject_edits_in_ranges(
|
|
buffer.clone(),
|
|
vec![Point::new(0, 0)..Point::new(100, 0)],
|
|
None,
|
|
cx,
|
|
);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
|
|
// File should still contain all the content
|
|
assert!(fs.is_file(path!("/dir/new_file").as_ref()).await);
|
|
|
|
let content = buffer.read_with(cx, |buffer, _| buffer.text());
|
|
assert_eq!(content, "ai content\nuser added this line");
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_reject_after_accepting_hunk_on_created_file(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path("dir/new_file", cx)
|
|
})
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
// AI creates file with initial content
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("ai content v1", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_ne!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
|
|
// User accepts the single hunk
|
|
action_log.update(cx, |log, cx| {
|
|
let buffer_range = Anchor::min_max_range_for_buffer(buffer.read(cx).remote_id());
|
|
log.keep_edits_in_range(buffer.clone(), buffer_range, None, cx)
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
assert!(fs.is_file(path!("/dir/new_file").as_ref()).await);
|
|
|
|
// AI modifies the file
|
|
cx.update(|cx| {
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("ai content v2", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_ne!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
|
|
// User rejects the hunk
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let (task, _) = log.reject_edits_in_ranges(
|
|
buffer.clone(),
|
|
vec![Anchor::min_max_range_for_buffer(
|
|
buffer.read(cx).remote_id(),
|
|
)],
|
|
None,
|
|
cx,
|
|
);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert!(fs.is_file(path!("/dir/new_file").as_ref()).await,);
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"ai content v1"
|
|
);
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_reject_edits_on_previously_accepted_created_file(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path("dir/new_file", cx)
|
|
})
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
// AI creates file with initial content
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("ai content v1", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
|
|
// User clicks "Accept All"
|
|
action_log.update(cx, |log, cx| log.keep_all_edits(None, cx));
|
|
cx.run_until_parked();
|
|
assert!(fs.is_file(path!("/dir/new_file").as_ref()).await);
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]); // Hunks are cleared
|
|
|
|
// AI modifies file again
|
|
cx.update(|cx| {
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("ai content v2", cx));
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
assert_ne!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
|
|
// User clicks "Reject All"
|
|
action_log
|
|
.update(cx, |log, cx| log.reject_all_edits(None, cx))
|
|
.await;
|
|
cx.run_until_parked();
|
|
assert!(fs.is_file(path!("/dir/new_file").as_ref()).await);
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"ai content v1"
|
|
);
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test(iterations = 100)]
|
|
async fn test_random_diffs(mut rng: StdRng, cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let operations = env::var("OPERATIONS")
|
|
.map(|i| i.parse().expect("invalid `OPERATIONS` variable"))
|
|
.unwrap_or(20);
|
|
|
|
let text = RandomCharIter::new(&mut rng).take(50).collect::<String>();
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": text})).await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
|
|
for _ in 0..operations {
|
|
match rng.random_range(0..100) {
|
|
0..25 => {
|
|
action_log.update(cx, |log, cx| {
|
|
let range = buffer.read(cx).random_byte_range(0, &mut rng);
|
|
log::info!("keeping edits in range {:?}", range);
|
|
log.keep_edits_in_range(buffer.clone(), range, None, cx)
|
|
});
|
|
}
|
|
25..50 => {
|
|
action_log
|
|
.update(cx, |log, cx| {
|
|
let range = buffer.read(cx).random_byte_range(0, &mut rng);
|
|
log::info!("rejecting edits in range {:?}", range);
|
|
let (task, _) =
|
|
log.reject_edits_in_ranges(buffer.clone(), vec![range], None, cx);
|
|
task
|
|
})
|
|
.await
|
|
.unwrap();
|
|
}
|
|
_ => {
|
|
let is_agent_edit = rng.random_bool(0.5);
|
|
if is_agent_edit {
|
|
log::info!("agent edit");
|
|
} else {
|
|
log::info!("user edit");
|
|
}
|
|
cx.update(|cx| {
|
|
buffer.update(cx, |buffer, cx| buffer.randomly_edit(&mut rng, 1, cx));
|
|
if is_agent_edit {
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
if rng.random_bool(0.2) {
|
|
quiesce(&action_log, &buffer, cx);
|
|
}
|
|
}
|
|
|
|
quiesce(&action_log, &buffer, cx);
|
|
|
|
fn quiesce(
|
|
action_log: &Entity<ActionLog>,
|
|
buffer: &Entity<Buffer>,
|
|
cx: &mut TestAppContext,
|
|
) {
|
|
log::info!("quiescing...");
|
|
cx.run_until_parked();
|
|
action_log.update(cx, |log, cx| {
|
|
let tracked_buffer = log.tracked_buffers.get(buffer).unwrap();
|
|
let mut old_text = tracked_buffer.diff_base.clone();
|
|
let new_text = buffer.read(cx).as_rope();
|
|
for edit in tracked_buffer.unreviewed_edits.edits() {
|
|
let old_start = old_text.point_to_offset(Point::new(edit.new.start, 0));
|
|
let old_end = old_text.point_to_offset(cmp::min(
|
|
Point::new(edit.new.start + edit.old_len(), 0),
|
|
old_text.max_point(),
|
|
));
|
|
old_text.replace(
|
|
old_start..old_end,
|
|
&new_text.slice_rows(edit.new.clone()).to_string(),
|
|
);
|
|
}
|
|
pretty_assertions::assert_eq!(old_text.to_string(), new_text.to_string());
|
|
})
|
|
}
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_keep_edits_on_commit(cx: &mut gpui::TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.background_executor.clone());
|
|
fs.insert_tree(
|
|
path!("/project"),
|
|
json!({
|
|
".git": {},
|
|
"file.txt": "a\nb\nc\nd\ne\nf\ng\nh\ni\nj",
|
|
}),
|
|
)
|
|
.await;
|
|
fs.set_head_for_repo(
|
|
path!("/project/.git").as_ref(),
|
|
&[("file.txt", "a\nb\nc\nd\ne\nf\ng\nh\ni\nj".into())],
|
|
"0000000",
|
|
);
|
|
cx.run_until_parked();
|
|
|
|
let project = Project::test(fs.clone(), [path!("/project").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path(path!("/project/file.txt"), cx)
|
|
})
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.edit(
|
|
[
|
|
// Edit at the very start: a -> A
|
|
(Point::new(0, 0)..Point::new(0, 1), "A"),
|
|
// Deletion in the middle: remove lines d and e
|
|
(Point::new(3, 0)..Point::new(5, 0), ""),
|
|
// Modification: g -> GGG
|
|
(Point::new(6, 0)..Point::new(6, 1), "GGG"),
|
|
// Addition: insert new line after h
|
|
(Point::new(7, 1)..Point::new(7, 1), "\nNEW"),
|
|
// Edit the very last character: j -> J
|
|
(Point::new(9, 0)..Point::new(9, 1), "J"),
|
|
],
|
|
None,
|
|
cx,
|
|
);
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![
|
|
HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(1, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "a\n".into()
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(3, 0)..Point::new(3, 0),
|
|
diff_status: DiffHunkStatusKind::Deleted,
|
|
old_text: "d\ne\n".into()
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(4, 0)..Point::new(5, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "g\n".into()
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(6, 0)..Point::new(7, 0),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into()
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(8, 0)..Point::new(8, 1),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "j".into()
|
|
}
|
|
]
|
|
)]
|
|
);
|
|
|
|
// Simulate a git commit that matches some edits but not others:
|
|
// - Accepts the first edit (a -> A)
|
|
// - Accepts the deletion (remove d and e)
|
|
// - Makes a different change to g (g -> G instead of GGG)
|
|
// - Ignores the NEW line addition
|
|
// - Ignores the last line edit (j stays as j)
|
|
fs.set_head_for_repo(
|
|
path!("/project/.git").as_ref(),
|
|
&[("file.txt", "A\nb\nc\nf\nG\nh\ni\nj".into())],
|
|
"0000001",
|
|
);
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer.clone(),
|
|
vec![
|
|
HunkStatus {
|
|
range: Point::new(4, 0)..Point::new(5, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "g\n".into()
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(6, 0)..Point::new(7, 0),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into()
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(8, 0)..Point::new(8, 1),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "j".into()
|
|
}
|
|
]
|
|
)]
|
|
);
|
|
|
|
// Make another commit that accepts the NEW line but with different content
|
|
fs.set_head_for_repo(
|
|
path!("/project/.git").as_ref(),
|
|
&[("file.txt", "A\nb\nc\nf\nGGG\nh\nDIFFERENT\ni\nj".into())],
|
|
"0000002",
|
|
);
|
|
cx.run_until_parked();
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![(
|
|
buffer,
|
|
vec![
|
|
HunkStatus {
|
|
range: Point::new(6, 0)..Point::new(7, 0),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into()
|
|
},
|
|
HunkStatus {
|
|
range: Point::new(8, 0)..Point::new(8, 1),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "j".into()
|
|
}
|
|
]
|
|
)]
|
|
);
|
|
|
|
// Final commit that accepts all remaining edits
|
|
fs.set_head_for_repo(
|
|
path!("/project/.git").as_ref(),
|
|
&[("file.txt", "A\nb\nc\nf\nGGG\nh\nNEW\ni\nJ".into())],
|
|
"0000003",
|
|
);
|
|
cx.run_until_parked();
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_keep_edits_on_commit_with_shifted_diff_boundaries(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let initial_text = indoc! {"
|
|
use crate::{Alpha, Beta};
|
|
|
|
fn keep() {
|
|
work();
|
|
}
|
|
|
|
fn remove() {
|
|
work();
|
|
}
|
|
|
|
fn after() {
|
|
work();
|
|
}
|
|
"};
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(
|
|
path!("/project"),
|
|
json!({
|
|
".git": {},
|
|
"file.rs": initial_text,
|
|
}),
|
|
)
|
|
.await;
|
|
fs.set_head_for_repo(
|
|
path!("/project/.git").as_ref(),
|
|
&[("file.rs", initial_text.into())],
|
|
"0000000",
|
|
);
|
|
cx.run_until_parked();
|
|
|
|
let project = Project::test(fs.clone(), [path!("/project").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path(path!("/project/file.rs"), cx)
|
|
})
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
let final_text = indoc! {"
|
|
use crate::{Alpha};
|
|
|
|
fn keep() {
|
|
work();
|
|
}
|
|
|
|
fn after() {
|
|
work();
|
|
}
|
|
"};
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.set_text(final_text, cx);
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
assert!(!unreviewed_hunks(&action_log, cx).is_empty());
|
|
|
|
fs.set_head_for_repo(
|
|
path!("/project/.git").as_ref(),
|
|
&[("file.rs", final_text.into())],
|
|
"0000001",
|
|
);
|
|
cx.run_until_parked();
|
|
|
|
assert_eq!(unreviewed_hunks(&action_log, cx), vec![]);
|
|
}
|
|
|
|
/// Regression test: when head_commit updates before the BufferDiff's base
|
|
/// text does, an intermediate DiffChanged (e.g. from a buffer-edit diff
|
|
/// recalculation) must NOT consume the commit signal. The subscription
|
|
/// should only fire once the base text itself has changed.
|
|
#[gpui::test]
|
|
async fn test_keep_edits_on_commit_with_stale_diff_changed(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(
|
|
path!("/project"),
|
|
json!({
|
|
".git": {},
|
|
"file.txt": "aaa\nbbb\nccc\nddd\neee",
|
|
}),
|
|
)
|
|
.await;
|
|
fs.set_head_for_repo(
|
|
path!("/project/.git").as_ref(),
|
|
&[("file.txt", "aaa\nbbb\nccc\nddd\neee".into())],
|
|
"0000000",
|
|
);
|
|
cx.run_until_parked();
|
|
|
|
let project = Project::test(fs.clone(), [path!("/project").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path(path!("/project/file.txt"), cx)
|
|
})
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
// Agent makes an edit: bbb -> BBB
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.edit([(Point::new(1, 0)..Point::new(1, 3), "BBB")], None, cx);
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
|
|
// Verify the edit is tracked
|
|
let hunks = unreviewed_hunks(&action_log, cx);
|
|
assert_eq!(hunks.len(), 1);
|
|
let hunk = &hunks[0].1;
|
|
assert_eq!(hunk.len(), 1);
|
|
assert_eq!(hunk[0].old_text, "bbb\n");
|
|
|
|
// Simulate the race condition: update only the HEAD SHA first,
|
|
// without changing the committed file contents. This is analogous
|
|
// to compute_snapshot updating head_commit before
|
|
// reload_buffer_diff_bases has loaded the new base text.
|
|
fs.with_git_state(path!("/project/.git").as_ref(), true, |state| {
|
|
state.refs.insert("HEAD".into(), "0000001".into());
|
|
})
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
|
|
// Make a user edit (on a different line) to trigger a buffer diff
|
|
// recalculation. This fires DiffChanged while the BufferDiff base
|
|
// text is still the OLD text. With the old head_commit-based
|
|
// subscription this would "consume" the commit detection.
|
|
cx.update(|cx| {
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.edit([(Point::new(3, 0)..Point::new(3, 3), "DDD")], None, cx);
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
|
|
// Now update the committed file contents to match the buffer
|
|
// (the agent edit was committed). Keep the same SHA so head_commit
|
|
// does NOT change again — this is the second half of the race.
|
|
{
|
|
use git::repository::repo_path;
|
|
fs.with_git_state(path!("/project/.git").as_ref(), true, |state| {
|
|
state
|
|
.head_contents
|
|
.insert(repo_path("file.txt"), "aaa\nBBB\nccc\nDDD\neee".into());
|
|
})
|
|
.unwrap();
|
|
}
|
|
cx.run_until_parked();
|
|
|
|
// The agent's edit (bbb -> BBB) should be accepted because the
|
|
// committed content now matches. Only the user edit (ddd -> DDD)
|
|
// should remain, but since the user edit is tracked as coming from
|
|
// the user (ChangeAuthor::User) it would have been rebased into
|
|
// the diff base already. So no unreviewed hunks should remain.
|
|
assert_eq!(
|
|
unreviewed_hunks(&action_log, cx),
|
|
vec![],
|
|
"agent edits should have been accepted after the base text update"
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_undo_last_reject(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(
|
|
path!("/dir"),
|
|
json!({
|
|
"file1": "abc\ndef\nghi"
|
|
}),
|
|
)
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file1", cx))
|
|
.unwrap();
|
|
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
// Track the buffer and make an agent edit
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit(
|
|
[(Point::new(1, 0)..Point::new(1, 3), "AGENT_EDIT")],
|
|
None,
|
|
cx,
|
|
)
|
|
.unwrap()
|
|
});
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
|
|
// Verify the agent edit is there
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\nAGENT_EDIT\nghi"
|
|
);
|
|
assert!(!unreviewed_hunks(&action_log, cx).is_empty());
|
|
|
|
// Reject all edits
|
|
action_log
|
|
.update(cx, |log, cx| log.reject_all_edits(None, cx))
|
|
.await;
|
|
cx.run_until_parked();
|
|
|
|
// Verify the buffer is back to original
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\ndef\nghi"
|
|
);
|
|
assert!(unreviewed_hunks(&action_log, cx).is_empty());
|
|
|
|
// Verify undo state is available
|
|
assert!(action_log.read_with(cx, |log, _| log.has_pending_undo()));
|
|
|
|
// Undo the reject
|
|
action_log
|
|
.update(cx, |log, cx| log.undo_last_reject(cx))
|
|
.await;
|
|
|
|
cx.run_until_parked();
|
|
|
|
// Verify the agent edit is restored
|
|
assert_eq!(
|
|
buffer.read_with(cx, |buffer, _| buffer.text()),
|
|
"abc\nAGENT_EDIT\nghi"
|
|
);
|
|
|
|
// Verify undo state is cleared
|
|
assert!(!action_log.read_with(cx, |log, _| log.has_pending_undo()));
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_linked_action_log_buffer_read(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "hello world"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let parent_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let child_log =
|
|
cx.new(|_| ActionLog::new(project.clone()).with_linked_action_log(parent_log.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
child_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
});
|
|
|
|
// Neither log considers the buffer stale immediately after reading it.
|
|
let child_stale = cx.read(|cx| {
|
|
child_log
|
|
.read(cx)
|
|
.stale_buffers(cx)
|
|
.cloned()
|
|
.collect::<Vec<_>>()
|
|
});
|
|
let parent_stale = cx.read(|cx| {
|
|
parent_log
|
|
.read(cx)
|
|
.stale_buffers(cx)
|
|
.cloned()
|
|
.collect::<Vec<_>>()
|
|
});
|
|
assert!(child_stale.is_empty());
|
|
assert!(parent_stale.is_empty());
|
|
|
|
// Simulate a user edit after the agent read the file.
|
|
cx.update(|cx| {
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.edit([(0..5, "goodbye")], None, cx).unwrap();
|
|
});
|
|
});
|
|
cx.run_until_parked();
|
|
|
|
// Both child and parent should see the buffer as stale because both tracked
|
|
// it at the pre-edit version via buffer_read forwarding.
|
|
let child_stale = cx.read(|cx| {
|
|
child_log
|
|
.read(cx)
|
|
.stale_buffers(cx)
|
|
.cloned()
|
|
.collect::<Vec<_>>()
|
|
});
|
|
let parent_stale = cx.read(|cx| {
|
|
parent_log
|
|
.read(cx)
|
|
.stale_buffers(cx)
|
|
.cloned()
|
|
.collect::<Vec<_>>()
|
|
});
|
|
assert_eq!(child_stale, vec![buffer.clone()]);
|
|
assert_eq!(parent_stale, vec![buffer]);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_linked_action_log_buffer_edited(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "abc\ndef\nghi"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let parent_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let child_log =
|
|
cx.new(|_| ActionLog::new(project.clone()).with_linked_action_log(parent_log.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
child_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer
|
|
.edit([(Point::new(1, 0)..Point::new(1, 3), "DEF")], None, cx)
|
|
.unwrap();
|
|
});
|
|
child_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
|
|
let expected_hunks = vec![(
|
|
buffer,
|
|
vec![HunkStatus {
|
|
range: Point::new(1, 0)..Point::new(2, 0),
|
|
diff_status: DiffHunkStatusKind::Modified,
|
|
old_text: "def\n".into(),
|
|
}],
|
|
)];
|
|
assert_eq!(
|
|
unreviewed_hunks(&child_log, cx),
|
|
expected_hunks,
|
|
"child should track the agent edit"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&parent_log, cx),
|
|
expected_hunks,
|
|
"parent should also track the agent edit via linked log forwarding"
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_linked_action_log_buffer_created(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({})).await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let parent_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let child_log =
|
|
cx.new(|_| ActionLog::new(project.clone()).with_linked_action_log(parent_log.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path("dir/new_file", cx)
|
|
})
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
child_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
buffer.update(cx, |buffer, cx| buffer.set_text("hello", cx));
|
|
child_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.save_buffer(buffer.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
|
|
let expected_hunks = vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 5),
|
|
diff_status: DiffHunkStatusKind::Added,
|
|
old_text: "".into(),
|
|
}],
|
|
)];
|
|
assert_eq!(
|
|
unreviewed_hunks(&child_log, cx),
|
|
expected_hunks,
|
|
"child should track the created file"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&parent_log, cx),
|
|
expected_hunks,
|
|
"parent should also track the created file via linked log forwarding"
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_linked_action_log_will_delete_buffer(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "hello\n"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let parent_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let child_log =
|
|
cx.new(|_| ActionLog::new(project.clone()).with_linked_action_log(parent_log.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path.clone(), cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
child_log.update(cx, |log, cx| log.will_delete_buffer(buffer.clone(), cx));
|
|
});
|
|
project
|
|
.update(cx, |project, cx| project.delete_file(file_path, false, cx))
|
|
.unwrap()
|
|
.await
|
|
.unwrap();
|
|
cx.run_until_parked();
|
|
|
|
let expected_hunks = vec![(
|
|
buffer.clone(),
|
|
vec![HunkStatus {
|
|
range: Point::new(0, 0)..Point::new(0, 0),
|
|
diff_status: DiffHunkStatusKind::Deleted,
|
|
old_text: "hello\n".into(),
|
|
}],
|
|
)];
|
|
assert_eq!(
|
|
unreviewed_hunks(&child_log, cx),
|
|
expected_hunks,
|
|
"child should track the deleted file"
|
|
);
|
|
assert_eq!(
|
|
unreviewed_hunks(&parent_log, cx),
|
|
expected_hunks,
|
|
"parent should also track the deleted file via linked log forwarding"
|
|
);
|
|
}
|
|
|
|
/// Simulates the subagent scenario: two child logs linked to the same parent, each
|
|
/// editing a different file. The parent accumulates all edits while each child
|
|
/// only sees its own.
|
|
#[gpui::test]
|
|
async fn test_linked_action_log_independent_tracking(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(
|
|
path!("/dir"),
|
|
json!({
|
|
"file_a": "content of a",
|
|
"file_b": "content of b",
|
|
}),
|
|
)
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let parent_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let child_log_1 =
|
|
cx.new(|_| ActionLog::new(project.clone()).with_linked_action_log(parent_log.clone()));
|
|
let child_log_2 =
|
|
cx.new(|_| ActionLog::new(project.clone()).with_linked_action_log(parent_log.clone()));
|
|
|
|
let file_a_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path("dir/file_a", cx)
|
|
})
|
|
.unwrap();
|
|
let file_b_path = project
|
|
.read_with(cx, |project, cx| {
|
|
project.find_project_path("dir/file_b", cx)
|
|
})
|
|
.unwrap();
|
|
let buffer_a = project
|
|
.update(cx, |project, cx| project.open_buffer(file_a_path, cx))
|
|
.await
|
|
.unwrap();
|
|
let buffer_b = project
|
|
.update(cx, |project, cx| project.open_buffer(file_b_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
cx.update(|cx| {
|
|
child_log_1.update(cx, |log, cx| log.buffer_read(buffer_a.clone(), cx));
|
|
buffer_a.update(cx, |buffer, cx| {
|
|
buffer.edit([(0..0, "MODIFIED: ")], None, cx).unwrap();
|
|
});
|
|
child_log_1.update(cx, |log, cx| log.buffer_edited(buffer_a.clone(), cx));
|
|
|
|
child_log_2.update(cx, |log, cx| log.buffer_read(buffer_b.clone(), cx));
|
|
buffer_b.update(cx, |buffer, cx| {
|
|
buffer.edit([(0..0, "MODIFIED: ")], None, cx).unwrap();
|
|
});
|
|
child_log_2.update(cx, |log, cx| log.buffer_edited(buffer_b.clone(), cx));
|
|
});
|
|
cx.run_until_parked();
|
|
|
|
let child_1_changed: Vec<_> = cx.read(|cx| {
|
|
child_log_1
|
|
.read(cx)
|
|
.changed_buffers(cx)
|
|
.into_keys()
|
|
.collect()
|
|
});
|
|
let child_2_changed: Vec<_> = cx.read(|cx| {
|
|
child_log_2
|
|
.read(cx)
|
|
.changed_buffers(cx)
|
|
.into_keys()
|
|
.collect()
|
|
});
|
|
let parent_changed: Vec<_> = cx.read(|cx| {
|
|
parent_log
|
|
.read(cx)
|
|
.changed_buffers(cx)
|
|
.into_keys()
|
|
.collect()
|
|
});
|
|
|
|
assert_eq!(
|
|
child_1_changed,
|
|
vec![buffer_a.clone()],
|
|
"child 1 should only track file_a"
|
|
);
|
|
assert_eq!(
|
|
child_2_changed,
|
|
vec![buffer_b.clone()],
|
|
"child 2 should only track file_b"
|
|
);
|
|
assert_eq!(parent_changed.len(), 2, "parent should track both files");
|
|
assert!(
|
|
parent_changed.contains(&buffer_a) && parent_changed.contains(&buffer_b),
|
|
"parent should contain both buffer_a and buffer_b"
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_file_read_time_recorded_on_buffer_read(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "hello world"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
let abs_path = PathBuf::from(path!("/dir/file"));
|
|
assert!(
|
|
action_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_none()),
|
|
"file_read_time should be None before buffer_read"
|
|
);
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
});
|
|
|
|
assert!(
|
|
action_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_some()),
|
|
"file_read_time should be recorded after buffer_read"
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_file_read_time_recorded_on_buffer_edited(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "hello world"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
let abs_path = PathBuf::from(path!("/dir/file"));
|
|
assert!(
|
|
action_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_none()),
|
|
"file_read_time should be None before buffer_edited"
|
|
);
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
|
|
assert!(
|
|
action_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_some()),
|
|
"file_read_time should be recorded after buffer_edited"
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_file_read_time_recorded_on_buffer_created(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "existing content"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
let abs_path = PathBuf::from(path!("/dir/file"));
|
|
assert!(
|
|
action_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_none()),
|
|
"file_read_time should be None before buffer_created"
|
|
);
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
});
|
|
|
|
assert!(
|
|
action_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_some()),
|
|
"file_read_time should be recorded after buffer_created"
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_file_read_time_removed_on_delete(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "hello world"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let action_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
let abs_path = PathBuf::from(path!("/dir/file"));
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
});
|
|
assert!(
|
|
action_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_some()),
|
|
"file_read_time should exist after buffer_read"
|
|
);
|
|
|
|
cx.update(|cx| {
|
|
action_log.update(cx, |log, cx| log.will_delete_buffer(buffer.clone(), cx));
|
|
});
|
|
assert!(
|
|
action_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_none()),
|
|
"file_read_time should be removed after will_delete_buffer"
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
async fn test_file_read_time_not_forwarded_to_linked_action_log(cx: &mut TestAppContext) {
|
|
init_test(cx);
|
|
|
|
let fs = FakeFs::new(cx.executor());
|
|
fs.insert_tree(path!("/dir"), json!({"file": "hello world"}))
|
|
.await;
|
|
let project = Project::test(fs.clone(), [path!("/dir").as_ref()], cx).await;
|
|
let parent_log = cx.new(|_| ActionLog::new(project.clone()));
|
|
let child_log =
|
|
cx.new(|_| ActionLog::new(project.clone()).with_linked_action_log(parent_log.clone()));
|
|
|
|
let file_path = project
|
|
.read_with(cx, |project, cx| project.find_project_path("dir/file", cx))
|
|
.unwrap();
|
|
let buffer = project
|
|
.update(cx, |project, cx| project.open_buffer(file_path, cx))
|
|
.await
|
|
.unwrap();
|
|
|
|
let abs_path = PathBuf::from(path!("/dir/file"));
|
|
|
|
cx.update(|cx| {
|
|
child_log.update(cx, |log, cx| log.buffer_read(buffer.clone(), cx));
|
|
});
|
|
assert!(
|
|
child_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_some()),
|
|
"child should record file_read_time on buffer_read"
|
|
);
|
|
assert!(
|
|
parent_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_none()),
|
|
"parent should NOT get file_read_time from child's buffer_read"
|
|
);
|
|
|
|
cx.update(|cx| {
|
|
child_log.update(cx, |log, cx| log.buffer_edited(buffer.clone(), cx));
|
|
});
|
|
assert!(
|
|
parent_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_none()),
|
|
"parent should NOT get file_read_time from child's buffer_edited"
|
|
);
|
|
|
|
cx.update(|cx| {
|
|
child_log.update(cx, |log, cx| log.buffer_created(buffer.clone(), cx));
|
|
});
|
|
assert!(
|
|
parent_log.read_with(cx, |log, _| log.file_read_time(&abs_path).is_none()),
|
|
"parent should NOT get file_read_time from child's buffer_created"
|
|
);
|
|
}
|
|
|
|
#[derive(Debug, PartialEq)]
|
|
struct HunkStatus {
|
|
range: Range<Point>,
|
|
diff_status: DiffHunkStatusKind,
|
|
old_text: String,
|
|
}
|
|
|
|
fn unreviewed_hunks(
|
|
action_log: &Entity<ActionLog>,
|
|
cx: &TestAppContext,
|
|
) -> Vec<(Entity<Buffer>, Vec<HunkStatus>)> {
|
|
cx.read(|cx| {
|
|
action_log
|
|
.read(cx)
|
|
.changed_buffers(cx)
|
|
.into_iter()
|
|
.map(|(buffer, diff)| {
|
|
let snapshot = buffer.read(cx).snapshot();
|
|
(
|
|
buffer,
|
|
diff.read(cx)
|
|
.snapshot(cx)
|
|
.hunks(&snapshot)
|
|
.map(|hunk| HunkStatus {
|
|
diff_status: hunk.status().kind,
|
|
range: hunk.range,
|
|
old_text: diff
|
|
.read(cx)
|
|
.base_text(cx)
|
|
.text_for_range(hunk.diff_base_byte_range)
|
|
.collect(),
|
|
})
|
|
.collect(),
|
|
)
|
|
})
|
|
.collect()
|
|
})
|
|
}
|
|
}
|