log: warn if the provided path looks like a revset

This commit is contained in:
Waleed Khan 2022-11-18 15:41:35 -08:00
parent 70182fbf7a
commit 94815a7cb5
5 changed files with 186 additions and 88 deletions

View file

@ -64,6 +64,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
* `jj git` subcommands now support credential helpers. * `jj git` subcommands now support credential helpers.
* `jj log` will warn if it appears that the provided path was meant to be a
revset.
### Fixed bugs ### Fixed bugs
* (#463) A bug in the export of branches to Git caused spurious conflicted * (#463) A bug in the export of branches to Git caused spurious conflicted

View file

@ -1049,6 +1049,10 @@ pub fn optimize(expression: Rc<RevsetExpression>) -> Rc<RevsetExpression> {
pub trait Revset<'repo> { pub trait Revset<'repo> {
// All revsets currently iterate in order of descending index position // All revsets currently iterate in order of descending index position
fn iter<'revset>(&'revset self) -> RevsetIterator<'revset, 'repo>; fn iter<'revset>(&'revset self) -> RevsetIterator<'revset, 'repo>;
fn is_empty(&self) -> bool {
self.iter().next().is_none()
}
} }
pub struct RevsetIterator<'revset, 'repo: 'revset> { pub struct RevsetIterator<'revset, 'repo: 'revset> {

View file

@ -53,6 +53,7 @@ use crate::formatter::Formatter;
use crate::templater::TemplateFormatter; use crate::templater::TemplateFormatter;
use crate::ui::{ColorChoice, Ui}; use crate::ui::{ColorChoice, Ui};
#[derive(Debug)]
pub enum CommandError { pub enum CommandError {
UserError { UserError {
message: String, message: String,

View file

@ -2023,10 +2023,12 @@ fn cmd_log(ui: &mut Ui, command: &CommandHelper, args: &LogArgs) -> Result<(), C
let workspace_id = workspace_command.workspace_id(); let workspace_id = workspace_command.workspace_id();
let checkout_id = repo.view().get_wc_commit_id(&workspace_id); let checkout_id = repo.view().get_wc_commit_id(&workspace_id);
let matcher = workspace_command.matcher_from_values(&args.paths)?; let matcher = workspace_command.matcher_from_values(&args.paths)?;
let mut revset = workspace_command.evaluate_revset(&revset_expression)?; let revset = workspace_command.evaluate_revset(&revset_expression)?;
if !args.paths.is_empty() { let revset = if !args.paths.is_empty() {
revset = revset::filter_by_diff(repo.as_repo_ref(), matcher.as_ref(), revset); revset::filter_by_diff(repo.as_repo_ref(), matcher.as_ref(), revset)
} } else {
revset
};
let store = repo.store(); let store = repo.store();
let diff_format = (args.patch || args.diff_format.git || args.diff_format.summary) let diff_format = (args.patch || args.diff_format.git || args.diff_format.summary)
@ -2042,94 +2044,115 @@ fn cmd_log(ui: &mut Ui, command: &CommandHelper, args: &LogArgs) -> Result<(), C
&template_string, &template_string,
); );
let mut formatter = ui.stdout_formatter(); {
let mut formatter = formatter.as_mut(); let mut formatter = ui.stdout_formatter();
formatter.add_label("log")?; let mut formatter = formatter.as_mut();
formatter.add_label("log")?;
if !args.no_graph { if !args.no_graph {
let mut graph = AsciiGraphDrawer::new(&mut formatter); let mut graph = AsciiGraphDrawer::new(&mut formatter);
let iter: Box<dyn Iterator<Item = (IndexEntry, Vec<RevsetGraphEdge>)>> = if args.reversed { let iter: Box<dyn Iterator<Item = (IndexEntry, Vec<RevsetGraphEdge>)>> =
Box::new(revset.iter().graph().reversed()) if args.reversed {
} else { Box::new(revset.iter().graph().reversed())
Box::new(revset.iter().graph())
};
for (index_entry, edges) in iter {
let mut graphlog_edges = vec![];
// TODO: Should we update RevsetGraphIterator to yield this flag instead of all
// the missing edges since we don't care about where they point here
// anyway?
let mut has_missing = false;
for edge in edges {
match edge.edge_type {
RevsetGraphEdgeType::Missing => {
has_missing = true;
}
RevsetGraphEdgeType::Direct => graphlog_edges.push(Edge::Present {
direct: true,
target: edge.target,
}),
RevsetGraphEdgeType::Indirect => graphlog_edges.push(Edge::Present {
direct: false,
target: edge.target,
}),
}
}
if has_missing {
graphlog_edges.push(Edge::Missing);
}
let mut buffer = vec![];
let commit_id = index_entry.commit_id();
let commit = store.get_commit(&commit_id)?;
let is_checkout = Some(&commit_id) == checkout_id;
{
let mut formatter = ui.new_formatter(&mut buffer);
if is_checkout {
formatter.with_label("working_copy", |formatter| {
template.format(&commit, formatter)
})?;
} else { } else {
template.format(&commit, formatter.as_mut())?; Box::new(revset.iter().graph())
};
for (index_entry, edges) in iter {
let mut graphlog_edges = vec![];
// TODO: Should we update RevsetGraphIterator to yield this flag instead of all
// the missing edges since we don't care about where they point here
// anyway?
let mut has_missing = false;
for edge in edges {
match edge.edge_type {
RevsetGraphEdgeType::Missing => {
has_missing = true;
}
RevsetGraphEdgeType::Direct => graphlog_edges.push(Edge::Present {
direct: true,
target: edge.target,
}),
RevsetGraphEdgeType::Indirect => graphlog_edges.push(Edge::Present {
direct: false,
target: edge.target,
}),
}
}
if has_missing {
graphlog_edges.push(Edge::Missing);
}
let mut buffer = vec![];
let commit_id = index_entry.commit_id();
let commit = store.get_commit(&commit_id)?;
let is_checkout = Some(&commit_id) == checkout_id;
{
let mut formatter = ui.new_formatter(&mut buffer);
if is_checkout {
formatter.with_label("working_copy", |formatter| {
template.format(&commit, formatter)
})?;
} else {
template.format(&commit, formatter.as_mut())?;
}
}
if !buffer.ends_with(b"\n") {
buffer.push(b'\n');
}
if let Some(diff_format) = diff_format {
let mut formatter = ui.new_formatter(&mut buffer);
show_patch(
formatter.as_mut(),
&workspace_command,
&commit,
matcher.as_ref(),
diff_format,
)?;
}
let node_symbol = if is_checkout { b"@" } else { b"o" };
graph.add_node(
&index_entry.position(),
&graphlog_edges,
node_symbol,
&buffer,
)?;
}
} else {
let iter: Box<dyn Iterator<Item = IndexEntry>> = if args.reversed {
Box::new(revset.iter().reversed())
} else {
Box::new(revset.iter())
};
for index_entry in iter {
let commit = store.get_commit(&index_entry.commit_id())?;
template.format(&commit, formatter)?;
if let Some(diff_format) = diff_format {
show_patch(
formatter,
&workspace_command,
&commit,
matcher.as_ref(),
diff_format,
)?;
} }
} }
if !buffer.ends_with(b"\n") {
buffer.push(b'\n');
}
if let Some(diff_format) = diff_format {
let mut formatter = ui.new_formatter(&mut buffer);
show_patch(
formatter.as_mut(),
&workspace_command,
&commit,
matcher.as_ref(),
diff_format,
)?;
}
let node_symbol = if is_checkout { b"@" } else { b"o" };
graph.add_node(
&index_entry.position(),
&graphlog_edges,
node_symbol,
&buffer,
)?;
} }
} else { }
let iter: Box<dyn Iterator<Item = IndexEntry>> = if args.reversed {
Box::new(revset.iter().reversed()) // Check to see if the user might have specified a path when they intended
} else { // to specify a revset.
Box::new(revset.iter()) if let (None, [only_path]) = (&args.revisions, args.paths.as_slice()) {
}; if only_path == "." && workspace_command.parse_file_path(only_path)?.is_root() {
for index_entry in iter { // For users of e.g. Mercurial, where `.` indicates the current commit.
let commit = store.get_commit(&index_entry.commit_id())?; ui.write_warn(&format!(
template.format(&commit, formatter)?; "warning: The argument {only_path:?} is being interpreted as a path, but this is \
if let Some(diff_format) = diff_format { often not useful because all non-empty commits touch '.'. If you meant to show \
show_patch( the working copy commit, pass -r '@' instead.\n"
formatter, ))?;
&workspace_command, } else if revset.is_empty() && revset::parse(only_path, None).is_ok() {
&commit, ui.write_warn(&format!(
matcher.as_ref(), "warning: The argument {only_path:?} is being interpreted as a path. To specify a \
diff_format, revset, pass -r {only_path:?} instead.\n"
)?; ))?;
}
} }
} }

View file

@ -12,7 +12,7 @@
// See the License for the specific language governing permissions and // See the License for the specific language governing permissions and
// limitations under the License. // limitations under the License.
use common::TestEnvironment; use common::{get_stderr_string, get_stdout_string, TestEnvironment};
pub mod common; pub mod common;
@ -217,6 +217,73 @@ fn test_log_filtered_by_path() {
"###); "###);
} }
#[test]
fn test_log_warn_path_might_be_revset() {
let test_env = TestEnvironment::default();
test_env.jj_cmd_success(test_env.env_root(), &["init", "repo", "--git"]);
let repo_path = test_env.env_root().join("repo");
std::fs::write(repo_path.join("file1"), "foo\n").unwrap();
// Don't warn if the file actually exists.
let assert = test_env
.jj_cmd(&repo_path, &["log", "file1", "-T", "description"])
.assert()
.success();
insta::assert_snapshot!(get_stdout_string(&assert), @r###"
@ (no description set)
~
"###);
insta::assert_snapshot!(get_stderr_string(&assert), @"");
// Warn for `jj log .` specifically, for former Mercurial users.
let assert = test_env
.jj_cmd(&repo_path, &["log", ".", "-T", "description"])
.assert()
.success();
insta::assert_snapshot!(get_stdout_string(&assert), @r###"
@ (no description set)
~
"###);
insta::assert_snapshot!(get_stderr_string(&assert), @r###"warning: The argument "." is being interpreted as a path, but this is often not useful because all non-empty commits touch '.'. If you meant to show the working copy commit, pass -r '@' instead."###);
// ...but checking `jj log .` makes sense in a subdirectory.
let subdir = repo_path.join("dir");
std::fs::create_dir_all(&subdir).unwrap();
let assert = test_env.jj_cmd(&subdir, &["log", "."]).assert().success();
insta::assert_snapshot!(get_stdout_string(&assert), @"");
insta::assert_snapshot!(get_stderr_string(&assert), @"");
// Warn for `jj log @` instead of `jj log -r @`.
let assert = test_env
.jj_cmd(&repo_path, &["log", "@", "-T", "description"])
.assert()
.success();
insta::assert_snapshot!(get_stdout_string(&assert), @"");
insta::assert_snapshot!(get_stderr_string(&assert), @r###"
warning: The argument "@" is being interpreted as a path. To specify a revset, pass -r "@" instead.
"###);
// Warn when there's no path with the provided name.
let assert = test_env
.jj_cmd(&repo_path, &["log", "file2", "-T", "description"])
.assert()
.success();
insta::assert_snapshot!(get_stdout_string(&assert), @"");
insta::assert_snapshot!(get_stderr_string(&assert), @r###"
warning: The argument "file2" is being interpreted as a path. To specify a revset, pass -r "file2" instead.
"###);
// If an explicit revision is provided, then suppress the warning.
let assert = test_env
.jj_cmd(&repo_path, &["log", "@", "-r", "@", "-T", "description"])
.assert()
.success();
insta::assert_snapshot!(get_stdout_string(&assert), @"");
insta::assert_snapshot!(get_stderr_string(&assert), @r###"
"###);
}
#[test] #[test]
fn test_default_revset() { fn test_default_revset() {
let test_env = TestEnvironment::default(); let test_env = TestEnvironment::default();