Use the global settings todo keywords when parsing headlines.
This commit is contained in:
parent
70fafd801e
commit
028aeb70aa
12
src/main.rs
12
src/main.rs
@ -111,11 +111,13 @@ fn run_parse_on_file<P: AsRef<Path>>(org_path: P) -> Result<(), Box<dyn std::err
|
|||||||
.ok_or("Should be contained inside a directory.")?;
|
.ok_or("Should be contained inside a directory.")?;
|
||||||
let org_contents = std::fs::read_to_string(org_path)?;
|
let org_contents = std::fs::read_to_string(org_path)?;
|
||||||
let org_contents = org_contents.as_str();
|
let org_contents = org_contents.as_str();
|
||||||
let global_settings = GlobalSettings {
|
let file_access_interface = LocalFileAccessInterface {
|
||||||
radio_targets: Vec::new(),
|
working_directory: Some(parent_directory.to_path_buf()),
|
||||||
file_access: &LocalFileAccessInterface {
|
};
|
||||||
working_directory: Some(parent_directory.to_path_buf()),
|
let global_settings = {
|
||||||
},
|
let mut global_settings = GlobalSettings::default();
|
||||||
|
global_settings.file_access = &file_access_interface;
|
||||||
|
global_settings
|
||||||
};
|
};
|
||||||
let rust_parsed = parse_with_settings(org_contents, &global_settings)?;
|
let rust_parsed = parse_with_settings(org_contents, &global_settings)?;
|
||||||
let org_sexp = emacs_parse_file_org_document(org_path)?;
|
let org_sexp = emacs_parse_file_org_document(org_path)?;
|
||||||
|
@ -125,12 +125,13 @@ fn document_org_source<'b, 'g, 'r, 's>(
|
|||||||
final_settings.extend(setup_file_settings);
|
final_settings.extend(setup_file_settings);
|
||||||
}
|
}
|
||||||
final_settings.extend(document_settings);
|
final_settings.extend(document_settings);
|
||||||
let new_settings = apply_in_buffer_settings(final_settings, context.get_global_settings()).map_err(|_err| {
|
let new_settings = apply_in_buffer_settings(final_settings, context.get_global_settings())
|
||||||
nom::Err::Error(CustomError::MyError(MyError(
|
.map_err(|_err| {
|
||||||
"TODO: make this take an owned string so I can dump err.to_string() into it."
|
nom::Err::Error(CustomError::MyError(MyError(
|
||||||
.into(),
|
"TODO: make this take an owned string so I can dump err.to_string() into it."
|
||||||
)))
|
.into(),
|
||||||
})?;
|
)))
|
||||||
|
})?;
|
||||||
let new_context = context.with_global_settings(&new_settings);
|
let new_context = context.with_global_settings(&new_settings);
|
||||||
let context = &new_context;
|
let context = &new_context;
|
||||||
|
|
||||||
@ -382,7 +383,6 @@ fn headline<'b, 'g, 'r, 's>(
|
|||||||
exit_matcher: &headline_title_end,
|
exit_matcher: &headline_title_end,
|
||||||
});
|
});
|
||||||
let parser_context = context.with_additional_node(&parser_context);
|
let parser_context = context.with_additional_node(&parser_context);
|
||||||
let standard_set_object_matcher = parser_with_context!(standard_set_object)(&parser_context);
|
|
||||||
|
|
||||||
let (
|
let (
|
||||||
remaining,
|
remaining,
|
||||||
@ -393,8 +393,11 @@ fn headline<'b, 'g, 'r, 's>(
|
|||||||
*star_count > parent_stars
|
*star_count > parent_stars
|
||||||
}),
|
}),
|
||||||
space1,
|
space1,
|
||||||
opt(tuple((heading_keyword, space1))),
|
opt(tuple((
|
||||||
many1(standard_set_object_matcher),
|
parser_with_context!(heading_keyword)(&parser_context),
|
||||||
|
space1,
|
||||||
|
))),
|
||||||
|
many1(parser_with_context!(standard_set_object)(&parser_context)),
|
||||||
opt(tuple((space0, tags))),
|
opt(tuple((space0, tags))),
|
||||||
space0,
|
space0,
|
||||||
alt((line_ending, eof)),
|
alt((line_ending, eof)),
|
||||||
@ -443,9 +446,34 @@ fn single_tag<'r, 's>(input: OrgSource<'s>) -> Res<OrgSource<'s>, OrgSource<'s>>
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[cfg_attr(feature = "tracing", tracing::instrument(ret, level = "debug"))]
|
#[cfg_attr(feature = "tracing", tracing::instrument(ret, level = "debug"))]
|
||||||
fn heading_keyword<'s>(input: OrgSource<'s>) -> Res<OrgSource<'s>, OrgSource<'s>> {
|
fn heading_keyword<'b, 'g, 'r, 's>(
|
||||||
// TODO: This should take into account the value of "#+TODO:" ref https://orgmode.org/manual/Per_002dfile-keywords.html and possibly the configurable variable org-todo-keywords ref https://orgmode.org/manual/Workflow-states.html. Case is significant.
|
context: RefContext<'b, 'g, 'r, 's>,
|
||||||
alt((tag("TODO"), tag("DONE")))(input)
|
input: OrgSource<'s>,
|
||||||
|
) -> Res<OrgSource<'s>, OrgSource<'s>> {
|
||||||
|
let global_settings = context.get_global_settings();
|
||||||
|
if global_settings.in_progress_todo_keywords.is_empty()
|
||||||
|
&& global_settings.complete_todo_keywords.is_empty()
|
||||||
|
{
|
||||||
|
alt((tag("TODO"), tag("DONE")))(input)
|
||||||
|
} else {
|
||||||
|
for todo_keyword in global_settings
|
||||||
|
.in_progress_todo_keywords
|
||||||
|
.iter()
|
||||||
|
.chain(global_settings.complete_todo_keywords.iter())
|
||||||
|
.map(String::as_str)
|
||||||
|
{
|
||||||
|
let result = tag::<_, _, CustomError<_>>(todo_keyword)(input);
|
||||||
|
match result {
|
||||||
|
Ok((remaining, ent)) => {
|
||||||
|
return Ok((remaining, ent));
|
||||||
|
}
|
||||||
|
Err(_) => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(nom::Err::Error(CustomError::MyError(MyError(
|
||||||
|
"NoTodoKeyword".into(),
|
||||||
|
))))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'s> Document<'s> {
|
impl<'s> Document<'s> {
|
||||||
|
@ -6,7 +6,7 @@ use nom::multi::many0;
|
|||||||
use nom::multi::many_till;
|
use nom::multi::many_till;
|
||||||
|
|
||||||
use super::keyword::filtered_keyword;
|
use super::keyword::filtered_keyword;
|
||||||
use super::keyword_todo::keyword_todo;
|
use super::keyword_todo::todo_keywords;
|
||||||
use super::OrgSource;
|
use super::OrgSource;
|
||||||
use crate::error::Res;
|
use crate::error::Res;
|
||||||
use crate::types::Keyword;
|
use crate::types::Keyword;
|
||||||
@ -56,7 +56,7 @@ pub fn apply_in_buffer_settings<'g, 's, 'sf>(
|
|||||||
|| kw.key.eq_ignore_ascii_case("typ_todo")
|
|| kw.key.eq_ignore_ascii_case("typ_todo")
|
||||||
}) {
|
}) {
|
||||||
let (_, (in_progress_words, complete_words)) =
|
let (_, (in_progress_words, complete_words)) =
|
||||||
keyword_todo(kw.value).map_err(|err| err.to_string())?;
|
todo_keywords(kw.value).map_err(|err| err.to_string())?;
|
||||||
new_settings
|
new_settings
|
||||||
.in_progress_todo_keywords
|
.in_progress_todo_keywords
|
||||||
.extend(in_progress_words.into_iter().map(str::to_string));
|
.extend(in_progress_words.into_iter().map(str::to_string));
|
||||||
|
@ -12,14 +12,18 @@ use nom::sequence::tuple;
|
|||||||
|
|
||||||
use crate::error::Res;
|
use crate::error::Res;
|
||||||
|
|
||||||
|
// ref https://orgmode.org/manual/Per_002dfile-keywords.html
|
||||||
|
// ref https://orgmode.org/manual/Workflow-states.html
|
||||||
|
// Case is significant.
|
||||||
|
|
||||||
/// Parses the text in the value of a #+TODO keyword.
|
/// Parses the text in the value of a #+TODO keyword.
|
||||||
///
|
///
|
||||||
/// Example input: "foo bar baz | lorem ipsum"
|
/// Example input: "foo bar baz | lorem ipsum"
|
||||||
pub fn keyword_todo<'s>(input: &'s str) -> Res<&'s str, (Vec<&'s str>, Vec<&'s str>)> {
|
pub fn todo_keywords<'s>(input: &'s str) -> Res<&'s str, (Vec<&'s str>, Vec<&'s str>)> {
|
||||||
let (remaining, mut before_pipe_words) = separated_list0(space1, keyword_word)(input)?;
|
let (remaining, mut before_pipe_words) = separated_list0(space1, todo_keyword_word)(input)?;
|
||||||
let (remaining, after_pipe_words) = opt(tuple((
|
let (remaining, after_pipe_words) = opt(tuple((
|
||||||
tuple((space0, tag("|"), space0)),
|
tuple((space0, tag("|"), space0)),
|
||||||
separated_list0(space1, keyword_word),
|
separated_list0(space1, todo_keyword_word),
|
||||||
)))(remaining)?;
|
)))(remaining)?;
|
||||||
let (remaining, _eol) = alt((line_ending, eof))(remaining)?;
|
let (remaining, _eol) = alt((line_ending, eof))(remaining)?;
|
||||||
if let Some((_pipe, after_pipe_words)) = after_pipe_words {
|
if let Some((_pipe, after_pipe_words)) = after_pipe_words {
|
||||||
@ -39,7 +43,7 @@ pub fn keyword_todo<'s>(input: &'s str) -> Res<&'s str, (Vec<&'s str>, Vec<&'s s
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn keyword_word<'s>(input: &'s str) -> Res<&'s str, &'s str> {
|
fn todo_keyword_word<'s>(input: &'s str) -> Res<&'s str, &'s str> {
|
||||||
verify(take_till(|c| " \t\r\n|".contains(c)), |result: &str| {
|
verify(take_till(|c| " \t\r\n|".contains(c)), |result: &str| {
|
||||||
!result.is_empty()
|
!result.is_empty()
|
||||||
})(input)
|
})(input)
|
||||||
@ -51,7 +55,7 @@ mod tests {
|
|||||||
#[test]
|
#[test]
|
||||||
fn before_and_after() -> Result<(), Box<dyn std::error::Error>> {
|
fn before_and_after() -> Result<(), Box<dyn std::error::Error>> {
|
||||||
let input = "foo bar baz | lorem ipsum";
|
let input = "foo bar baz | lorem ipsum";
|
||||||
let (remaining, (before_pipe_words, after_pipe_words)) = keyword_todo(input)?;
|
let (remaining, (before_pipe_words, after_pipe_words)) = todo_keywords(input)?;
|
||||||
assert_eq!(remaining, "");
|
assert_eq!(remaining, "");
|
||||||
assert_eq!(before_pipe_words, vec!["foo", "bar", "baz"]);
|
assert_eq!(before_pipe_words, vec!["foo", "bar", "baz"]);
|
||||||
assert_eq!(after_pipe_words, vec!["lorem", "ipsum"]);
|
assert_eq!(after_pipe_words, vec!["lorem", "ipsum"]);
|
||||||
@ -61,7 +65,7 @@ mod tests {
|
|||||||
#[test]
|
#[test]
|
||||||
fn no_pipe() -> Result<(), Box<dyn std::error::Error>> {
|
fn no_pipe() -> Result<(), Box<dyn std::error::Error>> {
|
||||||
let input = "foo bar baz";
|
let input = "foo bar baz";
|
||||||
let (remaining, (before_pipe_words, after_pipe_words)) = keyword_todo(input)?;
|
let (remaining, (before_pipe_words, after_pipe_words)) = todo_keywords(input)?;
|
||||||
assert_eq!(remaining, "");
|
assert_eq!(remaining, "");
|
||||||
assert_eq!(before_pipe_words, vec!["foo", "bar"]);
|
assert_eq!(before_pipe_words, vec!["foo", "bar"]);
|
||||||
assert_eq!(after_pipe_words, vec!["baz"]);
|
assert_eq!(after_pipe_words, vec!["baz"]);
|
||||||
@ -71,7 +75,7 @@ mod tests {
|
|||||||
#[test]
|
#[test]
|
||||||
fn early_pipe() -> Result<(), Box<dyn std::error::Error>> {
|
fn early_pipe() -> Result<(), Box<dyn std::error::Error>> {
|
||||||
let input = "| foo bar baz";
|
let input = "| foo bar baz";
|
||||||
let (remaining, (before_pipe_words, after_pipe_words)) = keyword_todo(input)?;
|
let (remaining, (before_pipe_words, after_pipe_words)) = todo_keywords(input)?;
|
||||||
assert_eq!(remaining, "");
|
assert_eq!(remaining, "");
|
||||||
assert_eq!(before_pipe_words, Vec::<&str>::new());
|
assert_eq!(before_pipe_words, Vec::<&str>::new());
|
||||||
assert_eq!(after_pipe_words, vec!["foo", "bar", "baz"]);
|
assert_eq!(after_pipe_words, vec!["foo", "bar", "baz"]);
|
||||||
@ -81,7 +85,7 @@ mod tests {
|
|||||||
#[test]
|
#[test]
|
||||||
fn late_pipe() -> Result<(), Box<dyn std::error::Error>> {
|
fn late_pipe() -> Result<(), Box<dyn std::error::Error>> {
|
||||||
let input = "foo bar baz |";
|
let input = "foo bar baz |";
|
||||||
let (remaining, (before_pipe_words, after_pipe_words)) = keyword_todo(input)?;
|
let (remaining, (before_pipe_words, after_pipe_words)) = todo_keywords(input)?;
|
||||||
assert_eq!(remaining, "");
|
assert_eq!(remaining, "");
|
||||||
assert_eq!(before_pipe_words, vec!["foo", "bar", "baz"]);
|
assert_eq!(before_pipe_words, vec!["foo", "bar", "baz"]);
|
||||||
assert_eq!(after_pipe_words, Vec::<&str>::new());
|
assert_eq!(after_pipe_words, Vec::<&str>::new());
|
||||||
|
Loading…
x
Reference in New Issue
Block a user