Function
action
v1.0
Things 3 JSON Import URL Action Button
An action button for converting Markdown-formatted task lists into Things 3 JSON format and generating import URLs. A link will be appended to the message.
Function ID
things_3_json_import_url_action_button
Creator
@jameslong
Downloads
117+

Function Content
python
"""
title: Things 3 JSON Import URL Generator
author: jameslong
license: MIT
version: 1.0
date: 2024-07-31
description: An action button for converting Markdown-formatted task lists into Things 3 JSON format and generating import URLs. A link will be appended to the message.
icon_url: 
required_open_webui_version: 0.3.9
"""

from pydantic import BaseModel, Field
from typing import Optional
import json
import urllib.parse
import re


class Action:
    class Valves(BaseModel):
        pass

    class UserValves(BaseModel):
        show_status: bool = Field(
            default=True, description="Show status of the action."
        )
        pass

    def __init__(self):
        self.valves = self.Valves()
        pass

    def create_things_url(self, markdown_list: str) -> str:
        lines = markdown_list.strip().split("\n")
        items = []
        current_heading = None
        current_items = []
        project_title = "Packing List"  # Default title

        def add_section():
            nonlocal current_heading, current_items
            if current_heading is not None:
                items.append(
                    {"type": "heading", "attributes": {"title": current_heading}}
                )
            items.extend(current_items)
            current_items = []

        checkbox_pattern = re.compile(r"^\s*-\s*\[([ xX])\]\s*(.+)$")

        for i, line in enumerate(lines):
            line = line.strip()
            if i == 0 and line.startswith("# "):
                project_title = line[2:].strip()
            elif line.startswith("##"):
                add_section()
                current_heading = line[2:].strip()
            else:
                checkbox_match = checkbox_pattern.match(line)
                if checkbox_match:
                    status, title = checkbox_match.groups()
                    completed = status.lower() == "x"
                    current_items.append(
                        {
                            "type": "to-do",
                            "attributes": {
                                "title": title.strip(),
                                "completed": completed,
                            },
                        }
                    )
                elif line.startswith("- "):
                    current_items.append(
                        {"type": "to-do", "attributes": {"title": line[2:].strip()}}
                    )

        add_section()  # Add the last section

        things_json = [
            {"type": "project", "attributes": {"title": project_title, "items": items}}
        ]

        json_string = json.dumps(things_json, separators=(",", ":"))
        encoded_json = urllib.parse.quote(json_string)
        things_url = f"things:///json?data={encoded_json}"
        return things_url

    async def action(
        self,
        body: dict,
        __user__=None,
        __event_emitter__=None,
        __event_call__=None,
    ) -> Optional[dict]:
        print(f"action:{__name__}")

        user_valves = __user__.get("valves")
        if not user_valves:
            user_valves = self.UserValves()

        if __event_emitter__:
            last_assistant_message = body["messages"][-1]

            if user_valves.show_status:
                await __event_emitter__(
                    {
                        "type": "status",
                        "data": {
                            "description": "Generating Things 3 URL",
                            "done": False,
                        },
                    }
                )

            try:
                things_url = self.create_things_url(last_assistant_message["content"])

                if user_valves.show_status:
                    await __event_emitter__(
                        {
                            "type": "status",
                            "data": {
                                "description": "Things 3 URL Generated",
                                "done": True,
                            },
                        }
                    )

                # Add a citation with the Things 3 URL
                await __event_emitter__(
                    {
                        "type": "message",
                        "data": {
                            "content": f"\n- [Import to Things 3]({things_url})\n"
                        },
                    }
                )

            except Exception as e:
                print(f"Error generating Things 3 URL: {str(e)}")
                if user_valves.show_status:
                    await __event_emitter__(
                        {
                            "type": "status",
                            "data": {
                                "description": "Error Generating Things 3 URL",
                                "done": True,
                            },
                        }
                    )

                    # Add a citation with the error message
                    await __event_emitter__(
                        {
                            "type": "citation",
                            "data": {
                                "source": {"name": "Error:generating Things 3 URL"},
                                "document": [str(e)],
                                "metadata": [
                                    {"source": "Things 3 JSON Import URL Generator"}
                                ],
                            },
                        }
                    )

        return None