Skip to content

Commit 99a5743

Browse files
Add :start=sha[...] filter
Change: start-filter
1 parent 6f667d4 commit 99a5743

File tree

5 files changed

+79
-0
lines changed

5 files changed

+79
-0
lines changed

src/filter/grammar.pest

+2
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ char = {
1818

1919
filter_spec = { (
2020
filter_group
21+
| filter_group_arg
2122
| filter_presub
2223
| filter_subdir
2324
| filter_nop
@@ -26,6 +27,7 @@ filter_spec = { (
2627
)+ }
2728

2829
filter_group = { CMD_START ~ cmd? ~ GROUP_START ~ compose ~ GROUP_END }
30+
filter_group_arg = { CMD_START ~ cmd ~ "=" ~ argument ~ GROUP_START ~ compose ~ GROUP_END }
2931
filter_subdir = { CMD_START ~ "/" ~ argument }
3032
filter_nop = { CMD_START ~ "/" }
3133
filter_presub = { CMD_START ~ ":" ~ argument }

src/filter/mod.rs

+13
Original file line numberDiff line numberDiff line change
@@ -98,6 +98,7 @@ enum Op {
9898
Fold,
9999
Paths,
100100
Squash(Option<std::collections::HashMap<git2::Oid, (String, String, String)>>),
101+
Start(git2::Oid, Filter),
101102
Linear,
102103

103104
RegexReplace(regex::Regex, String),
@@ -223,6 +224,9 @@ fn spec2(op: &Op) -> String {
223224
Op::Exclude(b) => {
224225
format!(":exclude[{}]", spec(*b))
225226
}
227+
Op::Start(id, b) => {
228+
format!(":start={}[{}]", id, spec(*b))
229+
}
226230
Op::Workspace(path) => {
227231
format!(":workspace={}", parse::quote(&path.to_string_lossy()))
228232
}
@@ -374,6 +378,14 @@ fn apply_to_commit2(
374378
))
375379
.transpose()
376380
}
381+
Op::Start(id,filter) => {
382+
if *id == commit.id() {
383+
return apply_to_commit2(&to_op(*filter), commit, transaction);
384+
} else {
385+
386+
return apply_to_commit2(&Op::Nop, commit, transaction);
387+
}
388+
},
377389
_ => {
378390
if let Some(oid) = transaction.get(filter, commit.id()) {
379391
return Ok(Some(oid));
@@ -619,6 +631,7 @@ fn apply2<'a>(
619631
Op::Squash(None) => Ok(tree),
620632
Op::Squash(Some(_)) => Err(josh_error("not applicable to tree")),
621633
Op::Linear => Ok(tree),
634+
Op::Start(_,_) => Err(josh_error("not applicable to tree")),
622635

623636
Op::RegexReplace(regex, replacement) => {
624637
tree::regex_replace(tree.id(), &regex, &replacement, transaction)

src/filter/opt.rs

+1
Original file line numberDiff line numberDiff line change
@@ -328,6 +328,7 @@ fn step(filter: Filter) -> Filter {
328328
Op::Prefix(path)
329329
}
330330
}
331+
Op::Start(id, filter) => Op::Start(id, step(filter)),
331332
Op::Compose(filters) if filters.is_empty() => Op::Empty,
332333
Op::Compose(filters) if filters.len() == 1 => to_op(filters[0]),
333334
Op::Compose(mut filters) => {

src/filter/parse.rs

+14
Original file line numberDiff line numberDiff line change
@@ -103,6 +103,20 @@ fn parse_item(pair: pest::iterators::Pair<Rule>) -> JoshResult<Op> {
103103
_ => Err(josh_error("parse_item: no match {:?}")),
104104
}
105105
}
106+
Rule::filter_group_arg => {
107+
let v: Vec<_> = pair.into_inner().map(|x| unquote(x.as_str())).collect();
108+
109+
match v.as_slice() {
110+
[cmd, arg, args] => {
111+
let g = parse_group(args)?;
112+
match *cmd {
113+
"start" => Ok(Op::Start(git2::Oid::from_str(arg)?, to_filter(Op::Compose(g)))),
114+
_ => Err(josh_error("parse_item: no match")),
115+
}
116+
}
117+
_ => Err(josh_error("parse_item: no match {:?}")),
118+
}
119+
}
106120
_ => Err(josh_error("parse_item: no match")),
107121
}
108122
}

tests/filter/start.t

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
$ export RUST_BACKTRACE=1
2+
$ git init -q 1> /dev/null
3+
4+
$ echo contents1 > file1
5+
$ git add .
6+
$ git commit -m "add file1" 1> /dev/null
7+
8+
$ git log --graph --pretty=%s
9+
* add file1
10+
11+
$ git checkout -b branch2
12+
Switched to a new branch 'branch2'
13+
14+
$ echo contents2 > file1
15+
$ git add .
16+
$ git commit -m "mod file1" 1> /dev/null
17+
18+
$ echo contents3 > file3
19+
$ git add .
20+
$ git commit -m "mod file3" 1> /dev/null
21+
22+
$ git checkout master
23+
Switched to branch 'master'
24+
25+
$ echo contents3 > file2
26+
$ git add .
27+
$ git commit -m "add file2" 1> /dev/null
28+
29+
$ git merge -q branch2 --no-ff
30+
31+
$ git log --graph --pretty=%H
32+
* 1d69b7d2651f744be3416f2ad526aeccefb99310
33+
|\
34+
| * 86871b8775ad3baca86484337d1072aa1d386f7e
35+
| * 975d4c4975912729482cc864d321c5196a969271
36+
* | e707f76bb6a1390f28b2162da5b5eb6933009070
37+
|/
38+
* 0b4cf6c9efbbda1eada39fa9c1d21d2525b027bb
39+
40+
$ josh-filter -s :start=0b4cf6c9efbbda1eada39fa9c1d21d2525b027bb[:prefix=x] --update refs/heads/filtered
41+
42+
$ git log --graph --decorate --pretty=%H refs/heads/filtered
43+
* 1d69b7d2651f744be3416f2ad526aeccefb99310
44+
|\
45+
| * 86871b8775ad3baca86484337d1072aa1d386f7e
46+
| * 975d4c4975912729482cc864d321c5196a969271
47+
* | e707f76bb6a1390f28b2162da5b5eb6933009070
48+
|/
49+
* 0b4cf6c9efbbda1eada39fa9c1d21d2525b027bb

0 commit comments

Comments
 (0)