[modules/todoist] - New module that connects to https://api.todoist.com and displays number of tasks due

This commit is contained in:
Duarte Figueiredo 2023-04-19 11:50:25 +01:00
parent 79081ebb4f
commit cad45ecd2c
4 changed files with 138 additions and 0 deletions

View file

@ -0,0 +1,77 @@
# pylint: disable=C0111,R0903
"""
Displays the of Todoist tasks that are due:
* https://developer.todoist.com/rest/v2/#get-active-tasks
Uses `xdg-open` or `x-www-browser` to open web-pages.
Requires the following library:
* requests
Errors:
if the Todoist get active tasks query failed, the shown value is `n/a`
Parameters:
* todoist.token: Todoist api token, you can get it in https://todoist.com/app/settings/integrations/developer.
* todoist.filter: a filter statement defined by Todoist (https://todoist.com/help/articles/introduction-to-filters), eg: "!assigned to: others & (Overdue | due: today)"
"""
import shutil
from typing import Final
import requests
import core.decorators
import core.input
import core.module
import core.widget
HOST_API: Final[str] = "https://api.todoist.com"
HOST_WEBSITE: Final[str] = "https://todoist.com/app/today"
TASKS_URL: Final[str] = f"{HOST_API}/rest/v2/tasks"
class Module(core.module.Module):
@core.decorators.every(minutes=5)
def __init__(self, config, theme):
super().__init__(config, theme, core.widget.Widget(self.todoist))
self.__user_id = None
self.background = True
self.__label = ""
token = self.parameter("token", "")
self.__filter = self.parameter("filter", "")
self.__requests = requests.Session()
self.__requests.headers.update({"Authorization": f"Bearer {token}"})
cmd = "xdg-open"
if not shutil.which(cmd):
cmd = "x-www-browser"
core.input.register(
self,
button=core.input.LEFT_MOUSE,
cmd=f"{cmd} {HOST_WEBSITE}",
)
def todoist(self, _):
return self.__label
def update(self):
try:
self.__label = self.__get_pending_tasks()
except Exception:
self.__label = "n/a"
def __get_pending_tasks(self) -> str:
params = {"filter": self.__filter} if self.__filter else None
response = self.__requests.get(TASKS_URL, params=params)
data = response.json()
return str(len(data))

BIN
screenshots/todoist.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 744 B

View file

@ -0,0 +1,58 @@
from unittest import TestCase, mock
import pytest
from requests import Session
import core.config
import core.widget
import modules.contrib.todoist
pytest.importorskip("requests")
def build_todoist_module(todoist_filter=None):
config = core.config.Config([
"-p",
f"todoist.filter={todoist_filter}" if todoist_filter else ""
])
return modules.contrib.todoist.Module(config=config, theme=None)
def mock_tasks_api_response():
res = mock.Mock()
res.json = lambda: [
{
"id": "-1",
"project_id": "-1"
},
{
"id": "-2",
"project_id": "-2"
}
]
res.status_code = 200
return res
class TestTodoistUnit(TestCase):
def test_load_module(self):
__import__("modules.contrib.todoist")
@mock.patch.object(Session, "get", return_value=mock_tasks_api_response())
def test_default_values(self, mock_get):
module = build_todoist_module()
module.update()
assert module.widgets()[0].full_text() == "2"
mock_get.assert_called_with('https://api.todoist.com/rest/v2/tasks', params=None)
@mock.patch.object(Session, "get", return_value=mock_tasks_api_response())
def test_custom_filter(self, mock_get):
module = build_todoist_module(todoist_filter="!assigned to: others & (Overdue | due: today)")
module.update()
assert module.widgets()[0].full_text() == "2"
mock_get.assert_called_with('https://api.todoist.com/rest/v2/tasks',
params={'filter': '!assigned to: others & (Overdue | due: today)'})

View file

@ -587,6 +587,9 @@
"wakatime": {
"prefix": "\uF017"
},
"todoist": {
"prefix": "\uF14A"
},
"deezer": {
"prefix": "  "
},