2022-06-17 18:13:11 +02:00
|
|
|
# Standard Packages
|
|
|
|
import datetime
|
|
|
|
|
|
|
|
# Internal Packages
|
2023-02-14 21:50:51 +01:00
|
|
|
from khoj.processor.org_mode import orgnode
|
2022-06-17 18:13:11 +02:00
|
|
|
|
|
|
|
|
|
|
|
# Test
|
2022-09-10 14:22:26 +02:00
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_parse_entry_with_no_headings(tmp_path):
|
|
|
|
"Test parsing of entry with minimal fields"
|
|
|
|
# Arrange
|
2023-02-17 17:04:26 +01:00
|
|
|
entry = f"""Body Line 1"""
|
2022-09-10 14:22:26 +02:00
|
|
|
orgfile = create_file(tmp_path, entry)
|
|
|
|
|
|
|
|
# Act
|
|
|
|
entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
assert len(entries) == 1
|
2023-02-17 17:04:26 +01:00
|
|
|
assert entries[0].heading == f"{orgfile}"
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].tags == list()
|
|
|
|
assert entries[0].body == "Body Line 1"
|
|
|
|
assert entries[0].priority == ""
|
2022-09-10 14:22:26 +02:00
|
|
|
assert entries[0].Property("ID") == ""
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].closed == ""
|
|
|
|
assert entries[0].scheduled == ""
|
|
|
|
assert entries[0].deadline == ""
|
2022-09-10 14:22:26 +02:00
|
|
|
|
|
|
|
|
2022-06-17 18:13:11 +02:00
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_parse_minimal_entry(tmp_path):
|
|
|
|
"Test parsing of entry with minimal fields"
|
|
|
|
# Arrange
|
2023-02-17 17:04:26 +01:00
|
|
|
entry = f"""
|
2022-06-17 18:13:11 +02:00
|
|
|
* Heading
|
2023-02-17 17:04:26 +01:00
|
|
|
Body Line 1"""
|
2022-06-17 18:13:11 +02:00
|
|
|
orgfile = create_file(tmp_path, entry)
|
|
|
|
|
|
|
|
# Act
|
|
|
|
entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
assert len(entries) == 1
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].heading == "Heading"
|
|
|
|
assert entries[0].tags == list()
|
|
|
|
assert entries[0].body == "Body Line 1"
|
|
|
|
assert entries[0].priority == ""
|
2022-06-17 18:13:11 +02:00
|
|
|
assert entries[0].Property("ID") == ""
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].closed == ""
|
|
|
|
assert entries[0].scheduled == ""
|
|
|
|
assert entries[0].deadline == ""
|
2022-06-17 18:13:11 +02:00
|
|
|
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_parse_complete_entry(tmp_path):
|
|
|
|
"Test parsing of entry with all important fields"
|
|
|
|
# Arrange
|
2023-02-17 17:04:26 +01:00
|
|
|
entry = f"""
|
2022-08-10 12:48:18 +02:00
|
|
|
*** DONE [#A] Heading :Tag1:TAG2:tag3:
|
2022-06-17 18:13:11 +02:00
|
|
|
CLOSED: [1984-04-01 Sun 12:00] SCHEDULED: <1984-04-01 Sun 09:00> DEADLINE: <1984-04-01 Sun>
|
|
|
|
:PROPERTIES:
|
|
|
|
:ID: 123-456-789-4234-1231
|
|
|
|
:END:
|
|
|
|
:LOGBOOK:
|
|
|
|
CLOCK: [1984-04-01 Sun 09:00]--[1984-04-01 Sun 12:00] => 3:00
|
|
|
|
- Clocked Log 1
|
|
|
|
:END:
|
|
|
|
Body Line 1
|
2023-02-17 17:04:26 +01:00
|
|
|
Body Line 2"""
|
2022-06-17 18:13:11 +02:00
|
|
|
orgfile = create_file(tmp_path, entry)
|
|
|
|
|
|
|
|
# Act
|
|
|
|
entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
assert len(entries) == 1
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].heading == "Heading"
|
|
|
|
assert entries[0].todo == "DONE"
|
|
|
|
assert entries[0].tags == ["Tag1", "TAG2", "tag3"]
|
|
|
|
assert entries[0].body == "- Clocked Log 1\nBody Line 1\nBody Line 2"
|
|
|
|
assert entries[0].priority == "A"
|
2022-06-17 18:13:11 +02:00
|
|
|
assert entries[0].Property("ID") == "id:123-456-789-4234-1231"
|
2023-02-17 17:04:26 +01:00
|
|
|
assert entries[0].closed == datetime.date(1984, 4, 1)
|
|
|
|
assert entries[0].scheduled == datetime.date(1984, 4, 1)
|
|
|
|
assert entries[0].deadline == datetime.date(1984, 4, 1)
|
|
|
|
assert entries[0].logbook == [(datetime.datetime(1984, 4, 1, 9, 0, 0), datetime.datetime(1984, 4, 1, 12, 0, 0))]
|
2022-06-17 18:13:11 +02:00
|
|
|
|
|
|
|
|
2022-09-11 14:54:26 +02:00
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_render_entry_with_property_drawer_and_empty_body(tmp_path):
|
|
|
|
"Render heading entry with property drawer"
|
|
|
|
# Arrange
|
2023-02-17 17:04:26 +01:00
|
|
|
entry_to_render = f"""
|
2022-09-11 14:54:26 +02:00
|
|
|
*** [#A] Heading1 :tag1:
|
|
|
|
:PROPERTIES:
|
|
|
|
:ID: 111-111-111-1111-1111
|
|
|
|
:END:
|
|
|
|
\t\r \n
|
2023-02-17 17:04:26 +01:00
|
|
|
"""
|
2022-09-11 14:54:26 +02:00
|
|
|
orgfile = create_file(tmp_path, entry_to_render)
|
|
|
|
|
2023-02-17 17:04:26 +01:00
|
|
|
expected_entry = f"""*** [#A] Heading1 :tag1:
|
2022-09-11 14:54:26 +02:00
|
|
|
:PROPERTIES:
|
|
|
|
:LINE: file:{orgfile}::2
|
|
|
|
:ID: id:111-111-111-1111-1111
|
|
|
|
:SOURCE: [[file:{orgfile}::*Heading1]]
|
|
|
|
:END:
|
2023-02-17 17:04:26 +01:00
|
|
|
"""
|
2022-09-11 14:54:26 +02:00
|
|
|
|
|
|
|
# Act
|
|
|
|
parsed_entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
2023-02-17 17:04:26 +01:00
|
|
|
assert f"{parsed_entries[0]}" == expected_entry
|
2022-09-11 14:54:26 +02:00
|
|
|
|
|
|
|
|
2022-06-17 18:13:11 +02:00
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_all_links_to_entry_rendered(tmp_path):
|
|
|
|
"Ensure all links to entry rendered in property drawer from entry"
|
|
|
|
# Arrange
|
2023-02-17 17:04:26 +01:00
|
|
|
entry = f"""
|
2022-06-17 18:13:11 +02:00
|
|
|
*** [#A] Heading :tag1:
|
|
|
|
:PROPERTIES:
|
|
|
|
:ID: 123-456-789-4234-1231
|
|
|
|
:END:
|
|
|
|
Body Line 1
|
|
|
|
*** Heading2
|
|
|
|
Body Line 2
|
2023-02-17 17:04:26 +01:00
|
|
|
"""
|
2022-06-17 18:13:11 +02:00
|
|
|
orgfile = create_file(tmp_path, entry)
|
|
|
|
|
|
|
|
# Act
|
|
|
|
entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
# SOURCE link rendered with Heading
|
2023-02-17 17:04:26 +01:00
|
|
|
assert f":SOURCE: [[file:{orgfile}::*{entries[0].heading}]]" in f"{entries[0]}"
|
2022-06-17 18:13:11 +02:00
|
|
|
# ID link rendered with ID
|
2023-02-17 17:04:26 +01:00
|
|
|
assert f":ID: id:123-456-789-4234-1231" in f"{entries[0]}"
|
2022-06-17 18:13:11 +02:00
|
|
|
# LINE link rendered with line number
|
2023-02-17 17:04:26 +01:00
|
|
|
assert f":LINE: file:{orgfile}::2" in f"{entries[0]}"
|
2022-06-17 18:13:11 +02:00
|
|
|
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_source_link_to_entry_escaped_for_rendering(tmp_path):
|
|
|
|
"Test SOURCE link renders with square brackets in filename, heading escaped for org-mode rendering"
|
|
|
|
# Arrange
|
2023-02-17 17:04:26 +01:00
|
|
|
entry = f"""
|
2022-06-17 18:13:11 +02:00
|
|
|
*** [#A] Heading[1] :tag1:
|
|
|
|
:PROPERTIES:
|
|
|
|
:ID: 123-456-789-4234-1231
|
|
|
|
:END:
|
2023-02-17 17:04:26 +01:00
|
|
|
Body Line 1"""
|
2022-06-17 18:13:11 +02:00
|
|
|
orgfile = create_file(tmp_path, entry, filename="test[1].org")
|
|
|
|
|
|
|
|
# Act
|
|
|
|
entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
assert len(entries) == 1
|
|
|
|
# parsed heading from entry
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].heading == "Heading[1]"
|
2022-06-17 18:13:11 +02:00
|
|
|
# ensure SOURCE link has square brackets in filename, heading escaped in rendered entries
|
2023-02-17 17:04:26 +01:00
|
|
|
escaped_orgfile = f"{orgfile}".replace("[1]", "\\[1\\]")
|
2023-02-18 00:12:43 +01:00
|
|
|
assert f":SOURCE: [[file:{escaped_orgfile}::*Heading\\[1\\]" in f"{entries[0]}"
|
2022-06-17 18:13:11 +02:00
|
|
|
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_parse_multiple_entries(tmp_path):
|
|
|
|
"Test parsing of multiple entries"
|
|
|
|
# Arrange
|
2023-02-17 17:04:26 +01:00
|
|
|
content = f"""
|
2022-08-10 12:48:18 +02:00
|
|
|
*** FAILED [#A] Heading1 :tag1:
|
2022-06-17 18:13:11 +02:00
|
|
|
CLOSED: [1984-04-01 Sun 12:00] SCHEDULED: <1984-04-01 Sun 09:00> DEADLINE: <1984-04-01 Sun>
|
|
|
|
:PROPERTIES:
|
|
|
|
:ID: 123-456-789-4234-0001
|
|
|
|
:END:
|
|
|
|
:LOGBOOK:
|
|
|
|
CLOCK: [1984-04-01 Sun 09:00]--[1984-04-01 Sun 12:00] => 3:00
|
|
|
|
- Clocked Log 1
|
|
|
|
:END:
|
|
|
|
Body 1
|
|
|
|
|
2022-08-10 12:48:18 +02:00
|
|
|
*** CANCELLED [#A] Heading2 :tag2:
|
2022-06-17 18:13:11 +02:00
|
|
|
CLOSED: [1984-04-02 Sun 12:00] SCHEDULED: <1984-04-02 Sun 09:00> DEADLINE: <1984-04-02 Sun>
|
|
|
|
:PROPERTIES:
|
|
|
|
:ID: 123-456-789-4234-0002
|
|
|
|
:END:
|
|
|
|
:LOGBOOK:
|
2022-07-20 22:15:30 +02:00
|
|
|
CLOCK: [1984-04-02 Mon 09:00]--[1984-04-02 Mon 12:00] => 3:00
|
2022-06-17 18:13:11 +02:00
|
|
|
- Clocked Log 2
|
|
|
|
:END:
|
|
|
|
Body 2
|
|
|
|
|
2023-02-17 17:04:26 +01:00
|
|
|
"""
|
2022-06-17 18:13:11 +02:00
|
|
|
orgfile = create_file(tmp_path, content)
|
|
|
|
|
|
|
|
# Act
|
|
|
|
entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
assert len(entries) == 2
|
|
|
|
for index, entry in enumerate(entries):
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entry.heading == f"Heading{index+1}"
|
|
|
|
assert entry.todo == "FAILED" if index == 0 else "CANCELLED"
|
|
|
|
assert entry.tags == [f"tag{index+1}"]
|
|
|
|
assert entry.body == f"- Clocked Log {index+1}\nBody {index+1}\n\n"
|
|
|
|
assert entry.priority == "A"
|
2022-06-17 18:13:11 +02:00
|
|
|
assert entry.Property("ID") == f"id:123-456-789-4234-000{index+1}"
|
2023-02-17 17:04:26 +01:00
|
|
|
assert entry.closed == datetime.date(1984, 4, index + 1)
|
|
|
|
assert entry.scheduled == datetime.date(1984, 4, index + 1)
|
|
|
|
assert entry.deadline == datetime.date(1984, 4, index + 1)
|
|
|
|
assert entry.logbook == [
|
|
|
|
(datetime.datetime(1984, 4, index + 1, 9, 0, 0), datetime.datetime(1984, 4, index + 1, 12, 0, 0))
|
|
|
|
]
|
2022-06-17 18:13:11 +02:00
|
|
|
|
|
|
|
|
2022-09-10 14:22:26 +02:00
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_parse_entry_with_empty_title(tmp_path):
|
|
|
|
"Test parsing of entry with minimal fields"
|
|
|
|
# Arrange
|
2023-02-17 19:07:59 +01:00
|
|
|
entry = f"""#+TITLE:
|
2023-02-17 17:04:26 +01:00
|
|
|
Body Line 1"""
|
2022-09-10 14:22:26 +02:00
|
|
|
orgfile = create_file(tmp_path, entry)
|
|
|
|
|
|
|
|
# Act
|
|
|
|
entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
assert len(entries) == 1
|
2023-02-17 17:04:26 +01:00
|
|
|
assert entries[0].heading == f"{orgfile}"
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].tags == list()
|
|
|
|
assert entries[0].body == "Body Line 1"
|
|
|
|
assert entries[0].priority == ""
|
2022-09-10 14:22:26 +02:00
|
|
|
assert entries[0].Property("ID") == ""
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].closed == ""
|
|
|
|
assert entries[0].scheduled == ""
|
|
|
|
assert entries[0].deadline == ""
|
2022-09-10 14:22:26 +02:00
|
|
|
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_parse_entry_with_title_and_no_headings(tmp_path):
|
|
|
|
"Test parsing of entry with minimal fields"
|
|
|
|
# Arrange
|
2023-02-17 17:04:26 +01:00
|
|
|
entry = f"""#+TITLE: test
|
|
|
|
Body Line 1"""
|
2022-09-10 14:22:26 +02:00
|
|
|
orgfile = create_file(tmp_path, entry)
|
|
|
|
|
|
|
|
# Act
|
|
|
|
entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
assert len(entries) == 1
|
2023-02-17 17:04:26 +01:00
|
|
|
assert entries[0].heading == "test"
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].tags == list()
|
|
|
|
assert entries[0].body == "Body Line 1"
|
|
|
|
assert entries[0].priority == ""
|
2022-09-10 14:22:26 +02:00
|
|
|
assert entries[0].Property("ID") == ""
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].closed == ""
|
|
|
|
assert entries[0].scheduled == ""
|
|
|
|
assert entries[0].deadline == ""
|
2022-09-10 14:22:26 +02:00
|
|
|
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------------------------------------------
|
|
|
|
def test_parse_entry_with_multiple_titles_and_no_headings(tmp_path):
|
|
|
|
"Test parsing of entry with minimal fields"
|
|
|
|
# Arrange
|
2023-02-17 19:07:59 +01:00
|
|
|
entry = f"""#+TITLE: title1
|
2022-09-10 14:22:26 +02:00
|
|
|
Body Line 1
|
2023-02-17 17:04:26 +01:00
|
|
|
#+TITLE: title2 """
|
2022-09-10 14:22:26 +02:00
|
|
|
orgfile = create_file(tmp_path, entry)
|
|
|
|
|
|
|
|
# Act
|
|
|
|
entries = orgnode.makelist(orgfile)
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
assert len(entries) == 1
|
2023-02-17 17:04:26 +01:00
|
|
|
assert entries[0].heading == "title1 title2"
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].tags == list()
|
|
|
|
assert entries[0].body == "Body Line 1\n"
|
|
|
|
assert entries[0].priority == ""
|
2022-09-10 14:22:26 +02:00
|
|
|
assert entries[0].Property("ID") == ""
|
2022-09-11 11:25:26 +02:00
|
|
|
assert entries[0].closed == ""
|
|
|
|
assert entries[0].scheduled == ""
|
|
|
|
assert entries[0].deadline == ""
|
2022-09-10 14:22:26 +02:00
|
|
|
|
|
|
|
|
2022-06-17 18:13:11 +02:00
|
|
|
# Helper Functions
|
|
|
|
def create_file(tmp_path, entry, filename="test.org"):
|
|
|
|
org_file = tmp_path / f"notes/{filename}"
|
|
|
|
org_file.parent.mkdir()
|
|
|
|
org_file.touch()
|
|
|
|
org_file.write_text(entry)
|
2022-09-10 14:22:26 +02:00
|
|
|
return org_file
|