You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1033 lines
36 KiB

# -*- encoding: utf-8 -*-
##############################################################################
#
# PyOrgMode, a python module for treating with orgfiles
# Copyright (C) 2010 Jonathan BISSON (bissonjonathan on the google thing).
# All Rights Reserved
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
##############################################################################
"""
The PyOrgMode class is able to read,modify and create orgfiles. The internal
representation of the file allows the use of orgfiles easily in your projects.
"""
import copy
import re
import time
class OrgDate:
"""Functions for date management"""
format = 0
TIMED = 1
DATED = 2
WEEKDAYED = 4
ACTIVE = 8
INACTIVE = 16
RANGED = 32
REPEAT = 64
CLOCKED = 128
# TODO: Timestamp with repeater interval
DICT_RE = {'start': '[[<]',
'end': '[]>]',
'date': r'([0-9]{4})-([0-9]{2})-([0-9]{2})(\s+([\w.]+))?',
'time': '([0-9]{2}):([0-9]{2})',
'clock': '([0-9]{1}):([0-9]{2})',
'repeat': r'[\+\.]{1,2}\d+[dwmy]'}
def __init__(self, value=None):
"""
Initialisation of an OrgDate element.
"""
self.set_value(value)
def parse_datetime(self, s):
"""
Parses an org-mode date time string.
Returns (timed, weekdayed, time_struct, repeat).
"""
search_re = r'r(?P<date>{date})(\s+(?P<time>{time}))?'.format(
**self.DICT_RE)
s = re.search(search_re, s)
weekdayed = (len(s.group('date').split()) > 1)
weekday_suffix = ""
if weekdayed is True:
weekday_suffix = s.group('date').split()[1]
formats = {
'timed_weekday': [True, '{0} {1} {2}', '%Y-%m-%d %a %H:%M'],
'timed': [True, '{0} {2}', '%Y-%m-%d %H:%M'],
'nottimed_weekday': [False, '{0} {1}', '%Y-%m-%d %a'],
'nottimed': [False, '{0}', '%Y-%m-%d'],
}
# We ignore weekdays (e.g. "Mon", "Tue") because a single org file
# could mix dates in many locales, e.g. if it was edited through
# many compters, each with a different language
PARSE_WEEKDAYS=False
if s.group('time'):
if weekday_suffix == "" or not PARSE_WEEKDAYS:
format_date = 'timed'
else:
format_date = 'timed_weekday'
else:
if weekday_suffix == "" or not PARSE_WEEKDAYS:
format_date = 'nottimed'
else:
format_date = 'nottimed_weekday'
return (formats[format_date][0], weekdayed,
time.strptime(
formats[format_date][1].format(s.group('date').split()[0],
weekday_suffix,
s.group('time')),
formats[format_date][2]))
def set_value(self, value):
"""
Setting the value of this element (automatic recognition of format)
"""
self.value = None # By default…
if value is None:
return
# Checking whether it is an active date-time or not
if value[0] == '<':
self.format |= self.ACTIVE
elif value[0] == '[':
self.format |= self.INACTIVE
# time range on a single day
search_re = (r'{start}(?P<date>{date})\s+(?P<time1>{time})'
'-(?P<time2>{time}){end}').format(**self.DICT_RE)
match = re.search(search_re, value)
if match:
timed, weekdayed, self.value = self.parse_datetime(
match.group('date') + ' ' + match.group('time1'))
if weekdayed:
self.format |= self.WEEKDAYED
timed, weekdayed, self.end = self.parse_datetime(
match.group('date') + ' ' + match.group('time2'))
self.format |= self.TIMED | self.DATED | self.RANGED
return
# date range over several days
search_re = (r'{start}(?P<date1>{date}(\s+{time})?){end}--'
r'{start}(?P<date2>{date}(\s+{time})?){end}').format(
**self.DICT_RE)
match = re.search(search_re, value)
if match:
timed, weekdayed, self.value = self.parse_datetime(
match.group('date1'))
if timed:
self.format |= self.TIMED
if weekdayed:
self.format |= self.WEEKDAYED
timed, weekdayed, self.end = self.parse_datetime(
match.group('date2'))
self.format |= self.DATED | self.RANGED
return
# single date with no range
search_re = (r'{start}(?P<datetime>{date}(\s+{time})?)' +
r'(\s+(?P<repeat>{repeat}))?{end}').format(
**self.DICT_RE)
match = re.search(search_re, value)
if match:
timed, weekdayed, self.value = self.parse_datetime(
match.group('datetime'))
if match.group('repeat'):
self.repeat = match.group('repeat')
self.format |= self.REPEAT
self.format |= self.DATED
if timed:
self.format |= self.TIMED
if weekdayed:
self.format |= self.WEEKDAYED
self.end = None
return
# clocked time
search_re = '(?P<clocked>{clock})'.format(**self.DICT_RE)
match = re.search(search_re, value)
if match:
self.value = value
self.format |= self.CLOCKED
def get_value(self):
"""
Get the timestamp as a text according to the format
"""
if self.value is None:
return ""
fmt_dict = {'time': '%H:%M'}
if self.format & self.ACTIVE:
fmt_dict['start'], fmt_dict['end'] = '<', '>'
else:
fmt_dict['start'], fmt_dict['end'] = '[', ']'
if self.format & self.WEEKDAYED:
fmt_dict['date'] = '%Y-%m-%d %a'
if self.format & self.CLOCKED:
fmt_dict['clock'] = "%H:%M"
elif not self.format & self.WEEKDAYED:
fmt_dict['date'] = '%Y-%m-%d'
if self.format & self.RANGED:
if self.value[:3] == self.end[:3]:
# range is between two times on a single day
assert self.format & self.TIMED
return (time.strftime(
'{start}{date} {time}-'.format(**fmt_dict), self.value) +
time.strftime('{time}{end}'.format(**fmt_dict),
self.end))
else:
# range is between two days
if self.format & self.TIMED:
return (time.strftime(
'{start}{date} {time}{end}--'.format(**fmt_dict),
self.value) +
time.strftime(
'{start}{date} {time}{end}'.format(**fmt_dict),
self.end))
else:
return (time.strftime(
'{start}{date}{end}--'.format(**fmt_dict),
self.value) +
time.strftime(
'{start}{date}{end}'.format(**fmt_dict),
self.end))
if self.format & self.CLOCKED:
# clocked time, return as is
return self.value
else: # non-ranged time
# Repeated
if self.format & self.REPEAT:
fmt_dict['repeat'] = ' ' + self.repeat
else:
fmt_dict['repeat'] = ''
if self.format & self.TIMED:
return time.strftime(
'{start}{date} {time}{repeat}{end}'.format(
**fmt_dict),
self.value)
else:
return time.strftime(
'{start}{date}{repeat}{end}'.format(**fmt_dict),
self.value)
class OrgPlugin:
"""
Generic class for all plugins
"""
def __init__(self):
""" Generic initialization """
self.treated = True
# By default, the plugin system stores the indentation before the
# treatment
self.keepindent = True
self.keepindent_value = ""
def treat(self, current, line):
"""This is a wrapper function for _treat. Asks the plugin if he can manage
this kind of line. Returns True if it can"""
self.treated = True
if self.keepindent:
# Keep a trace of the indentation
self.keepindent_value = line[0:len(line)-len(line.lstrip(" \t"))]
return self._treat(current, line.lstrip(" \t"))
else:
return self._treat(current, line)
def _treat(self, current, line):
"""This is the function used by the plugin for the management of
the line."""
self.treated = False
return current
def _append(self, current, element):
""" Internal function that adds to current. """
if self.keepindent and hasattr(element, "set_indent"):
element.set_indent(self.keepindent_value)
return current.append(element)
def close(self, current):
""" A wrapper function for closing the module. """
self.treated = False
return self._close(current)
def _close(self, current):
"""This is the function used by the plugin to close everything that have been
opened."""
self.treated = False
return current
class OrgElement:
"""
Generic class for all Elements excepted text and unrecognized ones
"""
def __init__(self):
self.content = []
self.parent = None
self.level = 0
self.indent = ""
def append(self, element):
# TODO Check validity
self.content.append(element)
# Check if the element got a parent attribute
# If so, we can have childrens into this element
if hasattr(element, "parent"):
element.parent = self
return element
def set_indent(self, indent):
""" Transfer the indentation from plugin to element. """
self.indent = indent
def output(self):
""" Wrapper for the text output. """
return self.indent+self._output()
def _output(self):
""" This is the function really used by the plugin. """
return ""
def __str__(self):
""" Used to return a text when called. """
return self.output()
class OrgTodo():
"""Describes an individual TODO item for use in agendas and TODO lists"""
def __init__(self, heading, todo_state,
scheduled=None, deadline=None,
tags=None, priority=None,
path=[0], node=None
):
self.heading = heading
self.todo_state = todo_state
self.scheduled = scheduled
self.deadline = deadline
self.tags = tags
self.priority = priority
self.node = node
def __str__(self):
string = self.todo_state + " " + self.heading
return string
class OrgClock(OrgPlugin):
"""Plugin for Clock elements"""
def __init__(self):
OrgPlugin.__init__(self)
self.regexp = re.compile(
r"(?:\s*)CLOCK:(?:\s*)((?:<|\[).*(?:>||\]))--\
((?:<|\[).*(?:>||\])).+=>\s*(.*)")
def _treat(self, current, line):
clocked = self.regexp.findall(line)
if clocked:
self._append(current,
self.Element(clocked[0][0],
clocked[0][1],
clocked[0][2]))
else:
self.treated = False
return current
class Element(OrgElement):
"""Clock is an element taking into account CLOCK elements"""
TYPE = "CLOCK_ELEMENT"
def __init__(self, start="", stop="", duration=""):
OrgElement.__init__(self)
self.start = OrgDate(start)
self.stop = OrgDate(stop)
self.duration = OrgDate(duration)
def _output(self):
"""Outputs the Clock element in text format
(e.g CLOCK: [2010-11-20 Sun 19:42]--[2010-11-20 Sun 20:14] => 0:32)
"""
return "CLOCK: " + self.start.get_value() + "--" + \
self.stop.get_value() + " => "+self.duration.get_value()+"\n"
class OrgSchedule(OrgPlugin):
"""Plugin for Schedule elements"""
# TODO: Need to find a better way to do this
def __init__(self):
OrgPlugin.__init__(self)
self.regexp_scheduled = re.compile(
r"SCHEDULED: ((<|\[).*?(>|\])(--(<|\[).*?(>|\]))?)")
self.regexp_deadline = re.compile(
r"DEADLINE: ((<|\[).*?(>|\])(--(<|\[).*?(>|\]))?)")
self.regexp_closed = re.compile(
r"CLOSED: ((<|\[).*?(>|\])(--(<|\[).*?(>|\]))?)")
def _treat(self, current, line):
scheduled = self.regexp_scheduled.findall(line)
deadline = self.regexp_deadline.findall(line)
closed = self.regexp_closed.findall(line)
if scheduled != []:
scheduled = scheduled[0][0]
if closed != []:
closed = closed[0][0]
if deadline != []:
deadline = deadline[0][0]
if scheduled or deadline or closed:
self._append(current,
self.Element(scheduled, deadline, closed))
else:
self.treated = False
return current
class Element(OrgElement):
"""Schedule is an element taking into account DEADLINE, SCHEDULED and CLOSED
parameters of elements"""
DEADLINE = 1
SCHEDULED = 2
CLOSED = 4
TYPE = "SCHEDULE_ELEMENT"
def __init__(self, scheduled=[], deadline=[], closed=[]):
OrgElement.__init__(self)
self.type = 0
if scheduled != []:
self.type = self.type | self.SCHEDULED
self.scheduled = OrgDate(scheduled)
if deadline != []:
self.type = self.type | self.DEADLINE
self.deadline = OrgDate(deadline)
if closed != []:
self.type = self.type | self.CLOSED
self.closed = OrgDate(closed)
def _output(self):
"""Outputs the Schedule element in text format (e.g SCHEDULED:
<2010-10-10 10:10>)"""
output = ""
if self.type & self.SCHEDULED:
output = output + "SCHEDULED: "+self.scheduled.get_value()+" "
if self.type & self.DEADLINE:
output = output + "DEADLINE: "+self.deadline.get_value()+" "
if self.type & self.CLOSED:
output = output + "CLOSED: "+self.closed.get_value()+" "
if output != "":
output = output.rstrip() + "\n"
return output
class OrgDrawer(OrgPlugin):
"""A Plugin for drawers"""
def __init__(self):
OrgPlugin.__init__(self)
self.regexp = re.compile(r"^(?:\s*?)(?::)(\S.*?)(?::)\s*(.*?)$")
def _treat(self, current, line):
drawer = self.regexp.search(line)
if isinstance(current, OrgDrawer.Element): # We are in a drawer
if drawer:
if drawer.group(1).upper() == "END": # Ending drawer
current = current.parent
elif drawer.group(2): # Adding a property
self._append(current,
self.Property(drawer.group(1),
drawer.group(2)))
else: # Adding text in drawer
self._append(current,
line.rstrip("\n"))
elif drawer: # Creating a drawer
current = self._append(current,
OrgDrawer.Element(drawer.group(1)))
else:
self.treated = False
return current
# It is a drawer, change the current also (even if not modified)
return current
class Element(OrgElement):
"""A Drawer object, containing properties and text"""
TYPE = "DRAWER_ELEMENT"
def __init__(self, name=""):
OrgElement.__init__(self)
self.name = name
def _output(self):
output = ":" + self.name + ":\n"
for element in self.content:
output = output + str(element) + "\n"
output = output + self.indent + ":END:\n"
return output
class Property(OrgElement):
"""A Property object, used in drawers."""
def __init__(self, name="", value=""):
OrgElement.__init__(self)
self.name = name
self.value = value
def _output(self):
"""Outputs the property in text format (e.g. :name: value)"""
return ":" + self.name + ": " + self.value
class OrgTable(OrgPlugin):
"""A plugin for table managment"""
def __init__(self):
OrgPlugin.__init__(self)
self.regexp = re.compile(r"^\s*\|")
def _treat(self, current, line):
table = self.regexp.match(line)
if table:
if not isinstance(current, self.Element):
current = current.append(self.Element())
current.append(line.rstrip().strip("|").split("|"))
else:
if isinstance(current, self.Element):
current = current.parent
self.treated = False
return current
class Element(OrgElement):
"""
A Table object
"""
TYPE = "TABLE_ELEMENT"
def __init__(self):
OrgElement.__init__(self)
def _output(self):
output = ""
for element in self.content:
output = output + "|"
for cell in element:
output = output + str(cell) + "|"
output = output + "\n"
return output
class OrgNode(OrgPlugin):
def __init__(self):
OrgPlugin.__init__(self)
self.todo_list = ['TODO']
self.done_list = ['DONE']
# If the line starts by an indent, it is not a node
self.keepindent = False
def _treat(self, current, line):
# Build regexp
regexp_string = r"^(\*+)\s*"
if self.todo_list:
separator = ""
re_todos = "("
for todo_keyword in self.todo_list + self.done_list:
re_todos += separator
separator = "|"
re_todos += todo_keyword
re_todos += r")?\s*"
regexp_string += re_todos
regexp_string += r"(\[#.*?\])?\s+(.*)$"
self.regexp = re.compile(regexp_string)
heading = self.regexp.findall(line)
if heading: # We have a heading
if current.parent:
current.parent.append(current)
# Is that a new level ?
if (len(heading[0][0]) > current.level): # Yes
# Parent is now the current node
parent = current
else:
# If not, the parent of the current node is the parent
parent = current.parent
# If we are going back one or more levels, walk through parents
while len(heading[0][0]) < current.level:
current = current.parent
parent = current.parent
# Creating a new node and assigning parameters
current = OrgNode.Element()
current.level = len(heading[0][0])
current.heading = re.sub(r":([:\w@]+)*:",
"",
heading[0][3]) # Remove tags
current.priority = heading[0][2].strip('[#]')
current.parent = parent
if heading[0][1]:
current.todo = heading[0][1]
# Looking for tags
heading_without_links = re.sub(r" \[(.+)\]", "", heading[0][3])
heading_without_title = re.sub(r"^(?:.+)\s+(?=:)", "",
heading_without_links)
matches = re.finditer(r'(?=:([\w@]+):)', heading_without_links)
# if no change, there is no residual string that
# follows the tag grammar
if heading_without_links != heading_without_title:
matches = re.finditer(r'(?=:([\w@]+):)',
heading_without_title)
[current.tags.append(match.group(1)) for match in matches]
else:
self.treated = False
return current
def _close(self, current):
# Add the last node
if (current.level > 0) and current.parent:
current.parent.append(current)
class Element(OrgElement):
# Defines an OrgMode Node in a structure
# The ID is auto-generated using uuid.
# The level 0 is the document itself
TYPE = "NODE_ELEMENT"
def __init__(self):
OrgElement.__init__(self)
self.content = []
self.level = 0
self.heading = ""
self.priority = ""
self.tags = []
# TODO Scheduling structure
def get_all_tags(self, use_tag_inheritance = True,
tags_exclude_from_inheritance = []):
"""Retrieve all tags applicable to this node, including those inherited from
parents or the document itself.
:param use_tag_inheritance: If None, the only tags that apply to a
node are those specified on it's
heading; if True, tags applied a
heading also apply to
sub-headings. Otherwise, it may also be
a list of tags that should be
inherited, or a regex matching tags to
be inherited.
:param tags_exclude_from_inheritance: gives an explicit list of
tags to be excluded from
inheritance, even if the
value of use_tag_inheritance
would select it for
inheritance.
Org nodes may have tags applied directly to them; this set is
contained in the ``tags`` attribute. Nodes may also *inherit* tags
from parents, or the file themselves (via the "#+FILETAGS"
in-buffer setting). Use this method to retrieve teh entire set
of tags that apply to this node, both direct & inherited.
These are analagous to the Emacs variables
``org-use-tag-inheritance`` and
``org-tags-exclude-from-inheritance``.
"""
retype = type(re.compile(''))
def _inheritable(tag):
if tag in tags_exclude_from_inheritance:
return False
if use_tag_inheritance == True:
return True
if use_tag_inheritance is None:
return False
if isinstance(use_tag_inheritance, retype):
return use_tag_inheritance.match(tag)
return tag in use_tag_inheritance
rv = copy.deepcopy(self.tags)
this = self
while this.parent is not None:
this = this.parent
rv.extend(filter(_inheritable, this.tags))
return rv
def _output(self):
output = ""
if hasattr(self, "level"):
output = output + "*"*self.level
if hasattr(self, "todo"):
output = output + " " + self.todo
if self.parent is not None:
output = output + " "
if self.priority:
output = output + "[#" + self.priority + "] "
output = output + self.heading
if self.tags:
output += ':' + ':'.join(self.tags) + ':'
output = output + "\n"
for element in self.content:
output = output + element.__str__()
return output
def append_clean(self, element):
if isinstance(element, list):
self.content.extend(element)
else:
self.content.append(element)
self.reparent_cleanlevels(self)
def reparent_cleanlevels(self, element=None, level=None):
"""
Reparent the childs elements of 'element' and make levels simpler.
Useful after moving one tree to another place or another file.
"""
if element is None:
element = self.root
if hasattr(element, "level"):
if level is None:
level = element.level
else:
element.level = level
if hasattr(element, "content"):
for child in element.content:
if hasattr(child, "parent"):
child.parent = element
self.reparent_cleanlevels(child,
level+1)
class OrgFileTags(OrgPlugin):
"""A plugin that recognizes FILETAGS & adds them to the root object."""
IN_BUFFER_RE = re.compile(r'\s*#\+FILETAGS:\s+([:a-zA-Z_]+)', re.I)
TAGS_RE = re.compile(r'(?=(^|:)(\w+)(:|$))', re.I)
def __init__(self):
OrgPlugin.__init__(self)
def _treat(self, current, line):
what = self.IN_BUFFER_RE.search(line)
if what is None:
self.treated = False
return current
file_tags = [m.group(2) for m in re.finditer(self.TAGS_RE, what.group(1))]
root = current
while root.parent is not None:
root = root.parent
root.tags.extend(file_tags)
self.treated = True
return current
class OrgDataStructure(OrgElement):
"""
Data structure containing all the nodes
The root property contains a reference to the level 0 node
"""
root = None
TYPE = "DATASTRUCTURE_ELEMENT"
def __init__(self):
OrgElement.__init__(self)
self.plugins = []
self.load_plugins(OrgTable(),
OrgDrawer(),
OrgNode(),
OrgSchedule(),
OrgClock(),
OrgFileTags())
# Add a root element
#
# The root node is a special node (no parent) used as a container for
# the file
self.root = OrgNode.Element()
self.root.parent = None
self.level = 0
def load_plugins(self, *arguments, **kw):
"""
Used to load plugins inside this DataStructure
"""
for plugin in arguments:
self.plugins.append(plugin)
def set_todo_states(self, new_states):
"""
Used to override the default list of todo states for any
OrgNode plugins in this object's plugins list. Expects
a list[] of strings as its argument. The list can be split
by '|' entries into TODO items and DONE items. Anything after
a second '|' will not be processed and be returned.
Setting to an empty list will disable TODO checking.
"""
new_todo_states = []
new_done_states = []
num_lists = 1
# Process the first part of the list (delimited by '|')
for new_state in new_states:
if new_state == '|':
num_lists += 1
break
new_todo_states.append(new_state)
# Clean up the lists so far
if num_lists > 1:
new_states.remove('|')
for todo_state in new_todo_states:
new_states.remove(todo_state)
# Process the second part of the list (delimited by '|')
for new_state in new_states:
if new_state == '|':
num_lists += 1
break
new_done_states.append(new_state)
# Clean up the second list
if num_lists > 2:
new_states.remove('|')
for todo_state in new_done_states:
new_states.remove(todo_state)
# Write the relevant attributes
for plugin in self.plugins:
if plugin.__class__ == OrgNode:
plugin.todo_list = new_todo_states
plugin.done_list = new_done_states
if new_states:
return new_states # Return any leftovers
def get_todo_states(self, list_type="todo"):
"""
Returns a list of todo states. An empty list means that
instance of OrgNode has TODO checking disabled. The first argument
determines the list that is pulled ("todo"*, "done" or "all").
"""
all_states = []
for plugin in self.plugins:
if plugin.__class__ == OrgNode:
if plugin.todo_list and (list_type == "todo" or list_type ==
"all"):
all_states += plugin.todo_list
if plugin.done_list and (list_type == "done" or list_type ==
"all"):
all_states += plugin.done_list
return list(set(all_states))
def add_todo_state(self, new_state):
"""Appends a todo state to the list of todo states of any OrgNode plugins in
this objects plugins list. Expects a string as its argument.
"""
for plugin in self.plugins:
if plugin.__class__ == OrgNode:
plugin.todo_list.append(new_state)
def add_done_state(self, new_state):
"""Appends a todo state to the list of todo states of any OrgNode plugins in
this objects plugins list. Expects a string as its argument.
"""
for plugin in self.plugins:
if plugin.__class__ == OrgNode:
plugin.done_list.append(new_state)
def remove_todo_state(self, old_state):
"""
Remove a given todo state from both the todo list and the done list.
Returns True if the plugin was actually found.
"""
found = False
for plugin in self.plugins:
if plugin.__class__ == OrgNode:
while old_state in plugin.todo_list:
found = True
plugin.todo_list.remove(old_state)
while old_state in plugin.done_list:
found = True
plugin.done_list.remove(old_state)
return found
def extract_todo_list(self, todo_list=None):
"""Extract a list of headings with TODO states specified by the first
argument.
"""
if todo_list is None: # Set default
# Kludge to get around lack of self in function declarations
todo_list = self.get_todo_states()
else:
# Check to make sure all todo_list items are registered
# with the OrgNode plugin
for possible_state in todo_list:
if possible_state not in self.get_todo_states("all"):
raise ValueError(
"State " + possible_state
+ " not registered. See \
PyOrgMode.OrgDataStructure.add_todo_state.")
results_list = []
# Recursive function that steps through each node in current level,
# looking for TODO items and then calls itself to look for
# TODO items one level down.
def extract_from_level(content):
for node in content:
# Check if it's a TODO item and add to results
try:
current_todo = node.todo
except AttributeError:
pass
else: # Handle it
if current_todo in todo_list:
new_todo = OrgTodo(node.heading,
node.todo,
tags=node.tags,
priority=node.priority,
node=node)
results_list.append(new_todo)
# Now check if it has sub-headings
try:
next_content = node.content
except AttributeError:
pass
else: # Handle it
extract_from_level(next_content)
extract_from_level(self.root.content)
return results_list
def load_from_file(self, name, form="file"):
"""
Used to load an org-file inside this DataStructure
"""
current = self.root
# Determine content type and put in appropriate form
if form == "file":
content = open(name, 'r')
elif form == "string":
content = [tmp+"\n" for tmp in name.split("\n")]
else:
raise ValueError("Form \""+form+"\" not recognized")
for line in content:
for plugin in self.plugins:
current = plugin.treat(current, line)
if plugin.treated: # Plugin found something
treated = True
break
else:
treated = False
if not treated and line is not None:
# Nothing special, just content
current.append(line)
for plugin in self.plugins:
current = plugin.close(current)
if form == "file":
content.close()
def load_from_string(self, string):
"""A wrapper calling load_from_file but with a string instead of reading from
a file.
"""
self.load_from_file(string, "string")
def save_to_file(self, name, node=None):
"""
Used to save an org-file corresponding to this DataStructure
"""
with open(name, 'w') as output:
if node is None:
node = self.root
output.write(str(node))
@staticmethod
def parse_heading(heading):
heading = heading.strip()
r = re.compile(r'(.*)(?:\s+\[(\d+)/(\d+)\])(?:\s+)?')
m = r.match(heading)
if m:
return {'heading': m.group(1),
'todo_done': m.group(2),
'todo_total': m.group(3)}
else:
return {'heading': heading}
@staticmethod
def get_nodes_by_priority(node, priority, found_nodes=[]):
# print "start of get_nodes_by_priority"
# print " node instance type: %s" % node.__class__.__name__
if isinstance(node, OrgElement):
# print " node.heading: %s" % node.heading
try:
if node.todo and node.priority == priority:
found_nodes.append(node)
except AttributeError:
# TODO: This could be a Property. Handle it!
pass
for node in node.content:
OrgDataStructure.get_nodes_by_priority(node,
priority,
found_nodes)
return found_nodes
else:
return found_nodes
@staticmethod
def get_node_by_heading(node, heading, found_nodes=[]):
if isinstance(node, OrgElement):
try:
heading_dict = OrgDataStructure.parse_heading(node.heading)
if heading_dict['heading'] == heading.strip():
found_nodes.append(node)
except AttributeError:
# TODO: This could be a Property. Handle it!
pass
for node in node.content:
OrgDataStructure.get_node_by_heading(node,
heading,
found_nodes)
return found_nodes
else:
return found_nodes