Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| ca7dc10e92 | |||
| 94efc524c6 | |||
| 436f897711 | |||
| 9d6cea0453 | |||
| c9433a5e98 | |||
| 7d3c5c2ae1 | |||
| 51c76ebd24 | |||
| facf39e778 | |||
| 51cecf1f4e | |||
| 79e4eb4de0 | |||
| 70b40cb90b | |||
| 22c32a9f3d | |||
| c3791ce961 | |||
| 27e24e2a86 |
File diff suppressed because it is too large
Load Diff
@@ -2,12 +2,17 @@
|
|||||||
|
|
||||||
import sqlite3
|
import sqlite3
|
||||||
import json
|
import json
|
||||||
from datetime import datetime
|
import shutil
|
||||||
|
import os
|
||||||
|
from datetime import datetime, timedelta
|
||||||
from typing import Optional, List, Dict, Any
|
from typing import Optional, List, Dict, Any
|
||||||
from contextlib import contextmanager
|
from contextlib import contextmanager
|
||||||
|
|
||||||
from .config import DATABASE_URL, TODO_STATUS, PRIORITY_LEVELS
|
from .config import DATABASE_URL, TODO_STATUS, PRIORITY_LEVELS
|
||||||
|
|
||||||
|
# 备份目录
|
||||||
|
BACKUP_DIR = os.path.join(os.path.dirname(DATABASE_URL), 'backups')
|
||||||
|
|
||||||
|
|
||||||
class Database:
|
class Database:
|
||||||
"""SQLite数据库管理"""
|
"""SQLite数据库管理"""
|
||||||
@@ -80,6 +85,28 @@ class Database:
|
|||||||
)
|
)
|
||||||
""")
|
""")
|
||||||
|
|
||||||
|
# 邮箱表
|
||||||
|
cursor.execute("""
|
||||||
|
CREATE TABLE IF NOT EXISTS emails (
|
||||||
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||||
|
email TEXT NOT NULL UNIQUE,
|
||||||
|
name TEXT,
|
||||||
|
created_at TEXT NOT NULL
|
||||||
|
)
|
||||||
|
""")
|
||||||
|
|
||||||
|
# 邮件发送记录表
|
||||||
|
cursor.execute("""
|
||||||
|
CREATE TABLE IF NOT EXISTS email_logs (
|
||||||
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||||
|
item_id INTEGER NOT NULL,
|
||||||
|
email TEXT NOT NULL,
|
||||||
|
sent_at TEXT NOT NULL,
|
||||||
|
success INTEGER DEFAULT 1,
|
||||||
|
FOREIGN KEY (item_id) REFERENCES items(id) ON DELETE CASCADE
|
||||||
|
)
|
||||||
|
""")
|
||||||
|
|
||||||
# 创建索引
|
# 创建索引
|
||||||
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_type ON items(type)")
|
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_type ON items(type)")
|
||||||
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_status ON items(status)")
|
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_status ON items(status)")
|
||||||
@@ -177,6 +204,41 @@ class Database:
|
|||||||
|
|
||||||
return items
|
return items
|
||||||
|
|
||||||
|
def count_items(self, type: str = None, status: str = None, tag: str = None,
|
||||||
|
keyword: str = None) -> int:
|
||||||
|
"""计算符合条件的条目总数"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
|
||||||
|
query = "SELECT COUNT(DISTINCT i.id) as count FROM items i"
|
||||||
|
params = []
|
||||||
|
conditions = []
|
||||||
|
|
||||||
|
# 标签过滤需要JOIN
|
||||||
|
if tag:
|
||||||
|
query += " JOIN item_tags it ON i.id = it.item_id JOIN tags t ON it.tag_id = t.id"
|
||||||
|
conditions.append("t.name = ?")
|
||||||
|
params.append(tag)
|
||||||
|
|
||||||
|
if type:
|
||||||
|
conditions.append("i.type = ?")
|
||||||
|
params.append(type)
|
||||||
|
|
||||||
|
if status:
|
||||||
|
conditions.append("i.status = ?")
|
||||||
|
params.append(status)
|
||||||
|
|
||||||
|
if keyword:
|
||||||
|
conditions.append("(i.title LIKE ? OR i.content LIKE ? OR i.note LIKE ?)")
|
||||||
|
keyword_pattern = f"%{keyword}%"
|
||||||
|
params.extend([keyword_pattern, keyword_pattern, keyword_pattern])
|
||||||
|
|
||||||
|
if conditions:
|
||||||
|
query += " WHERE " + " AND ".join(conditions)
|
||||||
|
|
||||||
|
cursor.execute(query, params)
|
||||||
|
return cursor.fetchone()['count']
|
||||||
|
|
||||||
def update_item(self, item_id: int, **kwargs) -> bool:
|
def update_item(self, item_id: int, **kwargs) -> bool:
|
||||||
"""更新条目"""
|
"""更新条目"""
|
||||||
allowed_fields = ['type', 'title', 'content', 'url', 'source', 'status', 'priority', 'due_date', 'note']
|
allowed_fields = ['type', 'title', 'content', 'url', 'source', 'status', 'priority', 'due_date', 'note']
|
||||||
@@ -244,6 +306,18 @@ class Database:
|
|||||||
""")
|
""")
|
||||||
return [dict(row) for row in cursor.fetchall()]
|
return [dict(row) for row in cursor.fetchall()]
|
||||||
|
|
||||||
|
def update_tag(self, tag_id: int, name: str) -> bool:
|
||||||
|
"""更新标签名称"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
# 检查名称是否已存在(排除自己)
|
||||||
|
cursor.execute("SELECT id FROM tags WHERE name = ? AND id != ?", (name, tag_id))
|
||||||
|
if cursor.fetchone():
|
||||||
|
return False # 名称已存在
|
||||||
|
cursor.execute("UPDATE tags SET name = ? WHERE id = ?", (name, tag_id))
|
||||||
|
conn.commit()
|
||||||
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
def delete_tag(self, tag_id: int = None, name: str = None) -> bool:
|
def delete_tag(self, tag_id: int = None, name: str = None) -> bool:
|
||||||
"""删除标签"""
|
"""删除标签"""
|
||||||
with self.get_conn() as conn:
|
with self.get_conn() as conn:
|
||||||
@@ -290,6 +364,90 @@ class Database:
|
|||||||
""", (item_id,))
|
""", (item_id,))
|
||||||
return [row['name'] for row in cursor.fetchall()]
|
return [row['name'] for row in cursor.fetchall()]
|
||||||
|
|
||||||
|
# ============ Email 操作 ============
|
||||||
|
|
||||||
|
def create_email(self, email: str, name: str = None) -> int:
|
||||||
|
"""创建邮箱"""
|
||||||
|
now = datetime.now().isoformat()
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
try:
|
||||||
|
cursor.execute("INSERT INTO emails (email, name, created_at) VALUES (?, ?, ?)",
|
||||||
|
(email, name, now))
|
||||||
|
conn.commit()
|
||||||
|
return cursor.lastrowid
|
||||||
|
except sqlite3.IntegrityError:
|
||||||
|
# 邮箱已存在,返回已有ID
|
||||||
|
cursor.execute("SELECT id FROM emails WHERE email = ?", (email,))
|
||||||
|
return cursor.fetchone()['id']
|
||||||
|
|
||||||
|
def list_emails(self) -> List[Dict[str, Any]]:
|
||||||
|
"""列出所有邮箱"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("SELECT * FROM emails ORDER BY created_at DESC")
|
||||||
|
return [dict(row) for row in cursor.fetchall()]
|
||||||
|
|
||||||
|
def get_email(self, email_id: int) -> Optional[Dict[str, Any]]:
|
||||||
|
"""获取单个邮箱"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("SELECT * FROM emails WHERE id = ?", (email_id,))
|
||||||
|
row = cursor.fetchone()
|
||||||
|
return dict(row) if row else None
|
||||||
|
|
||||||
|
def update_email(self, email_id: int, email: str = None, name: str = None) -> bool:
|
||||||
|
"""更新邮箱"""
|
||||||
|
now = datetime.now().isoformat()
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
if email:
|
||||||
|
# 检查邮箱是否已存在(排除自己)
|
||||||
|
cursor.execute("SELECT id FROM emails WHERE email = ? AND id != ?", (email, email_id))
|
||||||
|
if cursor.fetchone():
|
||||||
|
return False # 邎箱已存在
|
||||||
|
if email and name:
|
||||||
|
cursor.execute("UPDATE emails SET email = ?, name = ? WHERE id = ?", (email, name, email_id))
|
||||||
|
elif email:
|
||||||
|
cursor.execute("UPDATE emails SET email = ? WHERE id = ?", (email, email_id))
|
||||||
|
elif name:
|
||||||
|
cursor.execute("UPDATE emails SET name = ? WHERE id = ?", (name, email_id))
|
||||||
|
conn.commit()
|
||||||
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
def delete_email(self, email_id: int) -> bool:
|
||||||
|
"""删除邮箱"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("DELETE FROM emails WHERE id = ?", (email_id,))
|
||||||
|
conn.commit()
|
||||||
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
# ============ 邮件日志 操作 ============
|
||||||
|
|
||||||
|
def log_email_send(self, item_id: int, email: str, success: bool = True) -> int:
|
||||||
|
"""记录邮件发送"""
|
||||||
|
now = datetime.now().isoformat()
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("""
|
||||||
|
INSERT INTO email_logs (item_id, email, sent_at, success)
|
||||||
|
VALUES (?, ?, ?, ?)
|
||||||
|
""", (item_id, email, now, 1 if success else 0))
|
||||||
|
conn.commit()
|
||||||
|
return cursor.lastrowid
|
||||||
|
|
||||||
|
def get_email_logs(self, item_id: int) -> List[Dict[str, Any]]:
|
||||||
|
"""获取收藏的邮件发送记录"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("""
|
||||||
|
SELECT * FROM email_logs
|
||||||
|
WHERE item_id = ?
|
||||||
|
ORDER BY sent_at DESC
|
||||||
|
""", (item_id,))
|
||||||
|
return [dict(row) for row in cursor.fetchall()]
|
||||||
|
|
||||||
def stats(self) -> Dict[str, Any]:
|
def stats(self) -> Dict[str, Any]:
|
||||||
"""获取统计信息"""
|
"""获取统计信息"""
|
||||||
self._ensure_init()
|
self._ensure_init()
|
||||||
@@ -316,6 +474,252 @@ class Database:
|
|||||||
|
|
||||||
return stats
|
return stats
|
||||||
|
|
||||||
|
# ============ 提醒相关 ============
|
||||||
|
|
||||||
|
def get_reminders(self) -> Dict[str, Any]:
|
||||||
|
"""获取提醒信息:即将到期和已过期的待办"""
|
||||||
|
self._ensure_init()
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
now = datetime.now()
|
||||||
|
|
||||||
|
reminders = {
|
||||||
|
'overdue': [], # 已过期
|
||||||
|
'due_today': [], # 今天到期
|
||||||
|
'due_soon': [] # 24小时内到期(不含今天)
|
||||||
|
}
|
||||||
|
|
||||||
|
# 查询未完成的待办(有截止日期的)
|
||||||
|
cursor.execute("""
|
||||||
|
SELECT * FROM items
|
||||||
|
WHERE type = 'todo'
|
||||||
|
AND status != 'completed'
|
||||||
|
AND due_date IS NOT NULL
|
||||||
|
AND due_date != ''
|
||||||
|
ORDER BY due_date ASC
|
||||||
|
""")
|
||||||
|
|
||||||
|
for row in cursor.fetchall():
|
||||||
|
item = dict(row)
|
||||||
|
item['tags'] = self._get_item_tags(conn, item['id'])
|
||||||
|
|
||||||
|
try:
|
||||||
|
due_date_str = item['due_date']
|
||||||
|
# 支持多种日期格式
|
||||||
|
if 'T' in due_date_str:
|
||||||
|
# ISO 格式:2026-04-16T14:30
|
||||||
|
due_date = datetime.strptime(due_date_str[:16], '%Y-%m-%dT%H:%M')
|
||||||
|
elif len(due_date_str) == 10:
|
||||||
|
# 只有日期:2026-04-16,视为当天 23:59:59
|
||||||
|
due_date = datetime.strptime(due_date_str, '%Y-%m-%d').replace(hour=23, minute=59, second=59)
|
||||||
|
else:
|
||||||
|
# 其他格式,尝试解析
|
||||||
|
due_date = datetime.strptime(due_date_str.split('.')[0], '%Y-%m-%dT%H:%M:%S')
|
||||||
|
|
||||||
|
# 计算距离到期的时间
|
||||||
|
time_left = due_date - now
|
||||||
|
|
||||||
|
if time_left.total_seconds() < 0:
|
||||||
|
# 已过期
|
||||||
|
days_overdue = abs(int(time_left.total_seconds() / 86400))
|
||||||
|
item['days_overdue'] = days_overdue
|
||||||
|
reminders['overdue'].append(item)
|
||||||
|
elif time_left.total_seconds() < 86400: # 24小时内
|
||||||
|
# 判断是今天还是明天
|
||||||
|
if due_date.date() == now.date():
|
||||||
|
reminders['due_today'].append(item)
|
||||||
|
else:
|
||||||
|
reminders['due_soon'].append(item)
|
||||||
|
elif due_date.date() == now.date():
|
||||||
|
# 今天到期(超过24小时的情况,比如现在凌晨,截止时间是晚上)
|
||||||
|
reminders['due_today'].append(item)
|
||||||
|
except (ValueError, AttributeError) as e:
|
||||||
|
# 日期格式错误,跳过
|
||||||
|
continue
|
||||||
|
|
||||||
|
# 统计总数
|
||||||
|
reminders['total'] = len(reminders['overdue']) + len(reminders['due_today']) + len(reminders['due_soon'])
|
||||||
|
|
||||||
|
return reminders
|
||||||
|
|
||||||
|
# ============ 备份操作 ============
|
||||||
|
|
||||||
|
def create_backup(self, manual: bool = False) -> Dict[str, Any]:
|
||||||
|
"""创建数据库备份"""
|
||||||
|
import os
|
||||||
|
|
||||||
|
# 确保备份目录存在
|
||||||
|
os.makedirs(BACKUP_DIR, exist_ok=True)
|
||||||
|
|
||||||
|
now = datetime.now()
|
||||||
|
backup_name = now.strftime('%Y-%m-%d_%H%M%S')
|
||||||
|
if manual:
|
||||||
|
backup_name += '_manual'
|
||||||
|
backup_path = os.path.join(BACKUP_DIR, f'{backup_name}.db')
|
||||||
|
|
||||||
|
# 复制数据库文件
|
||||||
|
shutil.copy2(self.db_path, backup_path)
|
||||||
|
|
||||||
|
# 获取备份信息
|
||||||
|
backup_info = {
|
||||||
|
'name': backup_name,
|
||||||
|
'path': backup_path,
|
||||||
|
'size': os.path.getsize(backup_path),
|
||||||
|
'created_at': now.isoformat(),
|
||||||
|
'manual': manual,
|
||||||
|
'is_first_of_month': now.day == 1
|
||||||
|
}
|
||||||
|
|
||||||
|
# 保存备份元数据
|
||||||
|
self._save_backup_meta(backup_info)
|
||||||
|
|
||||||
|
# 清理旧备份
|
||||||
|
self._cleanup_old_backups()
|
||||||
|
|
||||||
|
return backup_info
|
||||||
|
|
||||||
|
def list_backups(self) -> List[Dict[str, Any]]:
|
||||||
|
"""列出所有备份"""
|
||||||
|
import os
|
||||||
|
|
||||||
|
if not os.path.exists(BACKUP_DIR):
|
||||||
|
return []
|
||||||
|
|
||||||
|
# 读取备份元数据
|
||||||
|
meta_path = os.path.join(BACKUP_DIR, 'backup_meta.json')
|
||||||
|
if os.path.exists(meta_path):
|
||||||
|
with open(meta_path, 'r') as f:
|
||||||
|
backups = json.load(f)
|
||||||
|
else:
|
||||||
|
# 从文件重建元数据
|
||||||
|
backups = []
|
||||||
|
for f in os.listdir(BACKUP_DIR):
|
||||||
|
if f.endswith('.db'):
|
||||||
|
path = os.path.join(BACKUP_DIR, f)
|
||||||
|
backups.append({
|
||||||
|
'name': f.replace('.db', ''),
|
||||||
|
'path': path,
|
||||||
|
'size': os.path.getsize(path),
|
||||||
|
'created_at': datetime.fromtimestamp(os.path.getmtime(path)).isoformat(),
|
||||||
|
'manual': '_manual' in f,
|
||||||
|
'is_first_of_month': self._is_first_of_month_filename(f)
|
||||||
|
})
|
||||||
|
|
||||||
|
# 按时间倒序排列
|
||||||
|
backups.sort(key=lambda x: x['created_at'], reverse=True)
|
||||||
|
|
||||||
|
return backups
|
||||||
|
|
||||||
|
def restore_backup(self, backup_name: str) -> bool:
|
||||||
|
"""恢复备份"""
|
||||||
|
import os
|
||||||
|
|
||||||
|
backup_path = os.path.join(BACKUP_DIR, f'{backup_name}.db')
|
||||||
|
if not os.path.exists(backup_path):
|
||||||
|
return False
|
||||||
|
|
||||||
|
# 先备份当前数据库(以防万一)
|
||||||
|
current_backup = self.db_path + '.before_restore'
|
||||||
|
shutil.copy2(self.db_path, current_backup)
|
||||||
|
|
||||||
|
# 恢复备份
|
||||||
|
shutil.copy2(backup_path, self.db_path)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def delete_backup(self, backup_name: str) -> bool:
|
||||||
|
"""删除备份"""
|
||||||
|
import os
|
||||||
|
|
||||||
|
backup_path = os.path.join(BACKUP_DIR, f'{backup_name}.db')
|
||||||
|
if not os.path.exists(backup_path):
|
||||||
|
return False
|
||||||
|
|
||||||
|
os.remove(backup_path)
|
||||||
|
|
||||||
|
# 更新元数据
|
||||||
|
self._remove_backup_meta(backup_name)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def _save_backup_meta(self, backup_info: Dict[str, Any]):
|
||||||
|
"""保存备份元数据"""
|
||||||
|
import os
|
||||||
|
|
||||||
|
meta_path = os.path.join(BACKUP_DIR, 'backup_meta.json')
|
||||||
|
|
||||||
|
# 读取现有元数据
|
||||||
|
backups = []
|
||||||
|
if os.path.exists(meta_path):
|
||||||
|
with open(meta_path, 'r') as f:
|
||||||
|
backups = json.load(f)
|
||||||
|
|
||||||
|
# 添加新备份
|
||||||
|
backups.append(backup_info)
|
||||||
|
|
||||||
|
# 保存
|
||||||
|
with open(meta_path, 'w') as f:
|
||||||
|
json.dump(backups, f, indent=2)
|
||||||
|
|
||||||
|
def _remove_backup_meta(self, backup_name: str):
|
||||||
|
"""从元数据中删除备份"""
|
||||||
|
import os
|
||||||
|
|
||||||
|
meta_path = os.path.join(BACKUP_DIR, 'backup_meta.json')
|
||||||
|
if not os.path.exists(meta_path):
|
||||||
|
return
|
||||||
|
|
||||||
|
with open(meta_path, 'r') as f:
|
||||||
|
backups = json.load(f)
|
||||||
|
|
||||||
|
backups = [b for b in backups if b['name'] != backup_name]
|
||||||
|
|
||||||
|
with open(meta_path, 'w') as f:
|
||||||
|
json.dump(backups, f, indent=2)
|
||||||
|
|
||||||
|
def _is_first_of_month_filename(self, filename: str) -> bool:
|
||||||
|
"""判断是否是每月第一天的备份"""
|
||||||
|
# 格式:2026-04-01_040000.db 或 2026-05-01_...
|
||||||
|
try:
|
||||||
|
date_part = filename.split('_')[0]
|
||||||
|
day = int(date_part.split('-')[2])
|
||||||
|
return day == 1
|
||||||
|
except:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def _cleanup_old_backups(self):
|
||||||
|
"""清理旧备份:保留30天 + 每月第一天"""
|
||||||
|
import os
|
||||||
|
|
||||||
|
backups = self.list_backups()
|
||||||
|
now = datetime.now()
|
||||||
|
keep_paths = []
|
||||||
|
|
||||||
|
for backup in backups:
|
||||||
|
backup_date = datetime.fromisoformat(backup['created_at'])
|
||||||
|
days_old = (now - backup_date).days
|
||||||
|
|
||||||
|
# 保留条件:
|
||||||
|
# 1. 手动备份永久保留(最多10个)
|
||||||
|
# 2. 30天内的备份
|
||||||
|
# 3. 每月第一天的备份
|
||||||
|
|
||||||
|
if backup['manual']:
|
||||||
|
# 手动备份保留,但最多10个
|
||||||
|
manual_backups = [b for b in backups if b['manual']]
|
||||||
|
if manual_backups.index(backup) < 10:
|
||||||
|
keep_paths.append(backup['path'])
|
||||||
|
else:
|
||||||
|
self.delete_backup(backup['name'])
|
||||||
|
elif days_old <= 30:
|
||||||
|
keep_paths.append(backup['path'])
|
||||||
|
elif backup['is_first_of_month']:
|
||||||
|
# 每月第一天永久保留
|
||||||
|
keep_paths.append(backup['path'])
|
||||||
|
else:
|
||||||
|
# 删除
|
||||||
|
self.delete_backup(backup['name'])
|
||||||
|
|
||||||
|
|
||||||
# 全局数据库实例
|
# 全局数据库实例
|
||||||
db = Database()
|
db = Database()
|
||||||
0
data/xian_favor.db
Normal file
0
data/xian_favor.db
Normal file
402
logs/api.log
Normal file
402
logs/api.log
Normal file
@@ -0,0 +1,402 @@
|
|||||||
|
* Serving Flask app 'xian_favor.api'
|
||||||
|
* Debug mode: off
|
||||||
|
WARNING: This is a development server. Do not use it in a production deployment. Use a production WSGI server instead.
|
||||||
|
* Running on all addresses (0.0.0.0)
|
||||||
|
* Running on http://127.0.0.1:19014
|
||||||
|
* Running on http://192.168.2.17:19014
|
||||||
|
Press CTRL+C to quit
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:06:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:04] "GET /api/items?limit=2&sort_by=updated_at&sort_order=desc HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:07:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:08:15] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:08:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.8 - - [19/Apr/2026 10:09:33] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:09:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:10:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:11:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:12:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:13:16] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:13:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.8 - - [19/Apr/2026 10:14:33] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:38] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:14:44] "GET / HTTP/1.1" 200 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:14:44] "GET /api/stats HTTP/1.1" 200 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:14:44] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:14:44] "GET /api/tags HTTP/1.1" 200 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:14:44] "GET /api/items?limit=20&offset=0 HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:14:58] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:01] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:08] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:15:16] "GET /api/items?limit=20&offset=0&sort_by=updated_at&sort_order=desc HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:15:21] "GET /api/items?limit=20&offset=0&sort_order=desc HTTP/1.1" 200 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:15:24] "GET /api/items?limit=20&offset=0&sort_by=created_at&sort_order=desc HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:15:27] "GET /api/items?limit=20&offset=0&sort_by=updated_at&sort_order=desc HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:28] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:31] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:15:34] "GET /api/items?limit=20&offset=0&sort_order=desc HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:15:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:15:56] "GET /api/items/59 HTTP/1.1" 200 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:16:00] "GET /api/items/59 HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:29] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:32] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:16:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:17:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:46] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:18:56] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:06] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:16] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:17] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:26] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.8 - - [19/Apr/2026 10:19:33] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:36] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:19:44] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:47] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:19:58] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:08] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:19] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:29] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:39] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:49] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:20:59] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:21:09] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:21:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:21:19] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:21:29] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:21:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:21:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:21:39] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:21:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:21:49] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:22:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:23:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.8 - - [19/Apr/2026 10:24:37] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:24:44] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:24:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:25:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:26:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:27:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:28:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.8 - - [19/Apr/2026 10:29:37] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:29:44] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:29:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:30:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:31:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:32:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:33:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.8 - - [19/Apr/2026 10:34:37] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:34:44] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:34:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:35:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:36:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:37:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:38:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.8 - - [19/Apr/2026 10:39:37] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:39:44] "GET /api/reminders HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:39:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:40:05] "PUT /api/items/59 HTTP/1.1" 200 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:40:05] "GET /api/stats HTTP/1.1" 200 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:40:05] "GET /api/items?limit=20&offset=0&sort_order=desc HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:40:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:40:57] "GET /api/items/68 HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
192.168.2.10 - - [19/Apr/2026 10:41:01] "GET /api/items/68 HTTP/1.1" 200 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:40] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:48] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:41:50] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:42:00] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:42:10] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:42:18] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:42:20] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:42:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:42:30] "GET /api/health HTTP/1.1" 404 -
|
||||||
|
127.0.0.1 - - [19/Apr/2026 10:42:33] "GET /api/health HTTP/1.1" 404 -
|
||||||
1679
logs/app.log
Normal file
1679
logs/app.log
Normal file
File diff suppressed because it is too large
Load Diff
9709
logs/server.log
Normal file
9709
logs/server.log
Normal file
File diff suppressed because it is too large
Load Diff
BIN
xian_favor/__pycache__/__main__.cpython-310.pyc
Normal file
BIN
xian_favor/__pycache__/__main__.cpython-310.pyc
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
File diff suppressed because it is too large
Load Diff
262
xian_favor/db.py
262
xian_favor/db.py
@@ -60,6 +60,9 @@ class Database:
|
|||||||
due_date TEXT,
|
due_date TEXT,
|
||||||
note TEXT,
|
note TEXT,
|
||||||
is_starred INTEGER DEFAULT 0,
|
is_starred INTEGER DEFAULT 0,
|
||||||
|
views INTEGER DEFAULT 0,
|
||||||
|
is_deleted INTEGER DEFAULT 0,
|
||||||
|
deleted_at TEXT,
|
||||||
created_at TEXT NOT NULL,
|
created_at TEXT NOT NULL,
|
||||||
updated_at TEXT NOT NULL
|
updated_at TEXT NOT NULL
|
||||||
)
|
)
|
||||||
@@ -96,6 +99,26 @@ class Database:
|
|||||||
)
|
)
|
||||||
""")
|
""")
|
||||||
|
|
||||||
|
# 草稿表
|
||||||
|
cursor.execute("""
|
||||||
|
CREATE TABLE IF NOT EXISTS drafts (
|
||||||
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||||
|
type TEXT NOT NULL DEFAULT 'text',
|
||||||
|
title TEXT,
|
||||||
|
content TEXT,
|
||||||
|
url TEXT,
|
||||||
|
source TEXT,
|
||||||
|
status TEXT DEFAULT 'pending',
|
||||||
|
priority TEXT DEFAULT 'medium',
|
||||||
|
due_date TEXT,
|
||||||
|
note TEXT,
|
||||||
|
tags TEXT,
|
||||||
|
is_starred INTEGER DEFAULT 0,
|
||||||
|
created_at TEXT NOT NULL,
|
||||||
|
updated_at TEXT NOT NULL
|
||||||
|
)
|
||||||
|
""")
|
||||||
|
|
||||||
# 邮件发送记录表
|
# 邮件发送记录表
|
||||||
cursor.execute("""
|
cursor.execute("""
|
||||||
CREATE TABLE IF NOT EXISTS email_logs (
|
CREATE TABLE IF NOT EXISTS email_logs (
|
||||||
@@ -112,7 +135,6 @@ class Database:
|
|||||||
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_type ON items(type)")
|
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_type ON items(type)")
|
||||||
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_status ON items(status)")
|
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_status ON items(status)")
|
||||||
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_created ON items(created_at)")
|
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_created ON items(created_at)")
|
||||||
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_starred ON items(is_starred)")
|
|
||||||
cursor.execute("CREATE INDEX IF NOT EXISTS idx_item_tags_item ON item_tags(item_id)")
|
cursor.execute("CREATE INDEX IF NOT EXISTS idx_item_tags_item ON item_tags(item_id)")
|
||||||
cursor.execute("CREATE INDEX IF NOT EXISTS idx_item_tags_tag ON item_tags(tag_id)")
|
cursor.execute("CREATE INDEX IF NOT EXISTS idx_item_tags_tag ON item_tags(tag_id)")
|
||||||
|
|
||||||
@@ -121,8 +143,26 @@ class Database:
|
|||||||
cursor.execute("SELECT is_starred FROM items LIMIT 1")
|
cursor.execute("SELECT is_starred FROM items LIMIT 1")
|
||||||
except sqlite3.OperationalError:
|
except sqlite3.OperationalError:
|
||||||
cursor.execute("ALTER TABLE items ADD COLUMN is_starred INTEGER DEFAULT 0")
|
cursor.execute("ALTER TABLE items ADD COLUMN is_starred INTEGER DEFAULT 0")
|
||||||
|
|
||||||
|
# 创建 is_starred 索引(字段添加后再创建)
|
||||||
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_starred ON items(is_starred)")
|
cursor.execute("CREATE INDEX IF NOT EXISTS idx_items_starred ON items(is_starred)")
|
||||||
|
|
||||||
|
# 检查并添加 views 字段(兼容旧数据库)
|
||||||
|
try:
|
||||||
|
cursor.execute("SELECT views FROM items LIMIT 1")
|
||||||
|
except sqlite3.OperationalError:
|
||||||
|
cursor.execute("ALTER TABLE items ADD COLUMN views INTEGER DEFAULT 0")
|
||||||
|
|
||||||
|
# 检查并添加 is_deleted 和 deleted_at 字段(兼容旧数据库)
|
||||||
|
try:
|
||||||
|
cursor.execute("SELECT is_deleted FROM items LIMIT 1")
|
||||||
|
except sqlite3.OperationalError:
|
||||||
|
cursor.execute("ALTER TABLE items ADD COLUMN is_deleted INTEGER DEFAULT 0")
|
||||||
|
try:
|
||||||
|
cursor.execute("SELECT deleted_at FROM items LIMIT 1")
|
||||||
|
except sqlite3.OperationalError:
|
||||||
|
cursor.execute("ALTER TABLE items ADD COLUMN deleted_at TEXT")
|
||||||
|
|
||||||
conn.commit()
|
conn.commit()
|
||||||
|
|
||||||
# ============ Item 操作 ============
|
# ============ Item 操作 ============
|
||||||
@@ -170,8 +210,13 @@ class Database:
|
|||||||
return item
|
return item
|
||||||
|
|
||||||
def list_items(self, type: str = None, status: str = None, tag: str = None,
|
def list_items(self, type: str = None, status: str = None, tag: str = None,
|
||||||
keyword: str = None, starred: bool = None, limit: int = 50, offset: int = 0) -> List[Dict[str, Any]]:
|
keyword: str = None, starred: bool = None, sort_by: str = None,
|
||||||
"""列出条目"""
|
sort_order: str = None, limit: int = 50, offset: int = 0) -> List[Dict[str, Any]]:
|
||||||
|
"""列出条目
|
||||||
|
|
||||||
|
sort_by: created_at, updated_at
|
||||||
|
sort_order: desc, asc
|
||||||
|
"""
|
||||||
with self.get_conn() as conn:
|
with self.get_conn() as conn:
|
||||||
cursor = conn.cursor()
|
cursor = conn.cursor()
|
||||||
|
|
||||||
@@ -179,6 +224,9 @@ class Database:
|
|||||||
params = []
|
params = []
|
||||||
conditions = []
|
conditions = []
|
||||||
|
|
||||||
|
# 只显示未删除的数据
|
||||||
|
conditions.append("i.is_deleted = 0")
|
||||||
|
|
||||||
# 标签过滤需要JOIN
|
# 标签过滤需要JOIN
|
||||||
if tag:
|
if tag:
|
||||||
query += " JOIN item_tags it ON i.id = it.item_id JOIN tags t ON it.tag_id = t.id"
|
query += " JOIN item_tags it ON i.id = it.item_id JOIN tags t ON it.tag_id = t.id"
|
||||||
@@ -205,8 +253,31 @@ class Database:
|
|||||||
if conditions:
|
if conditions:
|
||||||
query += " WHERE " + " AND ".join(conditions)
|
query += " WHERE " + " AND ".join(conditions)
|
||||||
|
|
||||||
# 重点关注优先显示
|
# 排序逻辑
|
||||||
query += " ORDER BY i.is_starred DESC, i.created_at DESC LIMIT ? OFFSET ?"
|
if sort_by == 'updated_at':
|
||||||
|
order_field = 'i.updated_at'
|
||||||
|
elif sort_by == 'created_at':
|
||||||
|
order_field = 'i.created_at'
|
||||||
|
else:
|
||||||
|
# 默认:重点关注优先 + 创建时间降序
|
||||||
|
order_field = 'i.created_at'
|
||||||
|
|
||||||
|
order_dir = 'DESC' if (sort_order == 'asc' or sort_order is None) else 'ASC'
|
||||||
|
# 这里反转逻辑:用户选择"降序"时用DESC,选择"升序"时用ASC
|
||||||
|
|
||||||
|
if sort_order == 'asc':
|
||||||
|
order_dir = 'ASC'
|
||||||
|
elif sort_order == 'desc':
|
||||||
|
order_dir = 'DESC'
|
||||||
|
else:
|
||||||
|
order_dir = 'DESC' # 默认降序
|
||||||
|
|
||||||
|
# 如果有指定排序字段,按该字段排序;否则默认重点关注优先
|
||||||
|
if sort_by:
|
||||||
|
query += f" ORDER BY {order_field} {order_dir} LIMIT ? OFFSET ?"
|
||||||
|
else:
|
||||||
|
# 默认:重点关注优先,然后创建时间降序
|
||||||
|
query += f" ORDER BY i.is_starred DESC, i.created_at DESC LIMIT ? OFFSET ?"
|
||||||
params.extend([limit, offset])
|
params.extend([limit, offset])
|
||||||
|
|
||||||
cursor.execute(query, params)
|
cursor.execute(query, params)
|
||||||
@@ -262,6 +333,7 @@ class Database:
|
|||||||
allowed_fields = ['type', 'title', 'content', 'url', 'source', 'status', 'priority', 'due_date', 'note', 'is_starred']
|
allowed_fields = ['type', 'title', 'content', 'url', 'source', 'status', 'priority', 'due_date', 'note', 'is_starred']
|
||||||
update_fields = {k: v for k, v in kwargs.items() if k in allowed_fields}
|
update_fields = {k: v for k, v in kwargs.items() if k in allowed_fields}
|
||||||
|
|
||||||
|
# 只有 tags 变化也算有效更新
|
||||||
if not update_fields and 'tags' not in kwargs:
|
if not update_fields and 'tags' not in kwargs:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@@ -270,6 +342,11 @@ class Database:
|
|||||||
with self.get_conn() as conn:
|
with self.get_conn() as conn:
|
||||||
cursor = conn.cursor()
|
cursor = conn.cursor()
|
||||||
|
|
||||||
|
# 检查条目是否存在
|
||||||
|
cursor.execute("SELECT id FROM items WHERE id = ?", (item_id,))
|
||||||
|
if not cursor.fetchone():
|
||||||
|
return False
|
||||||
|
|
||||||
if update_fields:
|
if update_fields:
|
||||||
set_clause = ", ".join(f"{k} = ?" for k in update_fields.keys())
|
set_clause = ", ".join(f"{k} = ?" for k in update_fields.keys())
|
||||||
set_clause += ", updated_at = ?"
|
set_clause += ", updated_at = ?"
|
||||||
@@ -284,16 +361,76 @@ class Database:
|
|||||||
self._add_tags_to_item(conn, item_id, kwargs['tags'])
|
self._add_tags_to_item(conn, item_id, kwargs['tags'])
|
||||||
|
|
||||||
conn.commit()
|
conn.commit()
|
||||||
return cursor.rowcount > 0
|
return True
|
||||||
|
|
||||||
def delete_item(self, item_id: int) -> bool:
|
def delete_item(self, item_id: int) -> bool:
|
||||||
"""删除条目"""
|
"""删除条目(移动到回收站)"""
|
||||||
with self.get_conn() as conn:
|
with self.get_conn() as conn:
|
||||||
cursor = conn.cursor()
|
cursor = conn.cursor()
|
||||||
|
now = datetime.now().isoformat()
|
||||||
|
cursor.execute("UPDATE items SET is_deleted = 1, deleted_at = ? WHERE id = ?", (now, item_id))
|
||||||
|
conn.commit()
|
||||||
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
def list_trash(self, limit: int = 50, offset: int = 0) -> List[Dict[str, Any]]:
|
||||||
|
"""列出回收站数据"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("SELECT * FROM items WHERE is_deleted = 1 ORDER BY deleted_at DESC LIMIT ? OFFSET ?", (limit, offset))
|
||||||
|
items = []
|
||||||
|
for row in cursor.fetchall():
|
||||||
|
item = dict(row)
|
||||||
|
item['tags'] = self._get_item_tags(conn, item['id'])
|
||||||
|
items.append(item)
|
||||||
|
return items
|
||||||
|
|
||||||
|
def count_trash(self) -> int:
|
||||||
|
"""计算回收站数据总数"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("SELECT COUNT(*) as count FROM items WHERE is_deleted = 1")
|
||||||
|
return cursor.fetchone()['count']
|
||||||
|
|
||||||
|
def restore_item(self, item_id: int) -> bool:
|
||||||
|
"""从回收站恢复数据"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("UPDATE items SET is_deleted = 0, deleted_at = NULL WHERE id = ?", (item_id,))
|
||||||
|
conn.commit()
|
||||||
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
def delete_permanently(self, item_id: int) -> bool:
|
||||||
|
"""彻底删除数据(从数据库中删除)"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
# 删除标签关联
|
||||||
|
cursor.execute("DELETE FROM item_tags WHERE item_id = ?", (item_id,))
|
||||||
|
# 删除邮件发送记录
|
||||||
|
cursor.execute("DELETE FROM email_logs WHERE item_id = ?", (item_id,))
|
||||||
|
# 删除数据
|
||||||
cursor.execute("DELETE FROM items WHERE id = ?", (item_id,))
|
cursor.execute("DELETE FROM items WHERE id = ?", (item_id,))
|
||||||
conn.commit()
|
conn.commit()
|
||||||
return cursor.rowcount > 0
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
def empty_trash(self) -> int:
|
||||||
|
"""清空回收站"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
# 获取所有回收站数据ID
|
||||||
|
cursor.execute("SELECT id FROM items WHERE is_deleted = 1")
|
||||||
|
ids = [row['id'] for row in cursor.fetchall()]
|
||||||
|
|
||||||
|
# 删除所有关联数据
|
||||||
|
for item_id in ids:
|
||||||
|
cursor.execute("DELETE FROM item_tags WHERE item_id = ?", (item_id,))
|
||||||
|
cursor.execute("DELETE FROM email_logs WHERE item_id = ?", (item_id,))
|
||||||
|
|
||||||
|
# 删除所有回收站数据
|
||||||
|
cursor.execute("DELETE FROM items WHERE is_deleted = 1")
|
||||||
|
deleted_count = cursor.rowcount
|
||||||
|
conn.commit()
|
||||||
|
return deleted_count
|
||||||
|
|
||||||
def toggle_star(self, item_id: int) -> bool:
|
def toggle_star(self, item_id: int) -> bool:
|
||||||
"""切换重点关注状态"""
|
"""切换重点关注状态"""
|
||||||
with self.get_conn() as conn:
|
with self.get_conn() as conn:
|
||||||
@@ -319,6 +456,117 @@ class Database:
|
|||||||
conn.commit()
|
conn.commit()
|
||||||
return cursor.rowcount > 0
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
def increment_views(self, item_id: int) -> bool:
|
||||||
|
"""增加阅读数"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("UPDATE items SET views = views + 1 WHERE id = ?", (item_id,))
|
||||||
|
conn.commit()
|
||||||
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
# ============ Draft 草稿操作 ============
|
||||||
|
|
||||||
|
def save_draft(self, type: str = "text", title: str = None, content: str = None,
|
||||||
|
url: str = None, source: str = None, status: str = "pending",
|
||||||
|
priority: str = "medium", due_date: str = None, note: str = None,
|
||||||
|
tags: str = None, is_starred: bool = False) -> int:
|
||||||
|
"""保存草稿"""
|
||||||
|
self._ensure_init()
|
||||||
|
now = datetime.now().isoformat()
|
||||||
|
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("""
|
||||||
|
INSERT INTO drafts (type, title, content, url, source, status, priority, due_date, note, tags, is_starred, created_at, updated_at)
|
||||||
|
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
||||||
|
""", (type, title, content, url, source, status, priority, due_date, note, tags, 1 if is_starred else 0, now, now))
|
||||||
|
draft_id = cursor.lastrowid
|
||||||
|
conn.commit()
|
||||||
|
return draft_id
|
||||||
|
|
||||||
|
def update_draft(self, draft_id: int, **kwargs) -> bool:
|
||||||
|
"""更新草稿"""
|
||||||
|
allowed_fields = ['type', 'title', 'content', 'url', 'source', 'status', 'priority', 'due_date', 'note', 'tags', 'is_starred']
|
||||||
|
update_fields = {k: v for k, v in kwargs.items() if k in allowed_fields}
|
||||||
|
|
||||||
|
if not update_fields:
|
||||||
|
return False
|
||||||
|
|
||||||
|
now = datetime.now().isoformat()
|
||||||
|
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("SELECT id FROM drafts WHERE id = ?", (draft_id,))
|
||||||
|
if not cursor.fetchone():
|
||||||
|
return False
|
||||||
|
|
||||||
|
set_clause = ", ".join(f"{k} = ?" for k in update_fields.keys())
|
||||||
|
set_clause += ", updated_at = ?"
|
||||||
|
values = list(update_fields.values()) + [now, draft_id]
|
||||||
|
cursor.execute(f"UPDATE drafts SET {set_clause} WHERE id = ?", values)
|
||||||
|
conn.commit()
|
||||||
|
return True
|
||||||
|
|
||||||
|
def list_drafts(self, limit: int = 50, offset: int = 0) -> List[Dict[str, Any]]:
|
||||||
|
"""列出草稿"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("SELECT * FROM drafts ORDER BY updated_at DESC LIMIT ? OFFSET ?", (limit, offset))
|
||||||
|
return [dict(row) for row in cursor.fetchall()]
|
||||||
|
|
||||||
|
def count_drafts(self) -> int:
|
||||||
|
"""计算草稿总数"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("SELECT COUNT(*) as count FROM drafts")
|
||||||
|
return cursor.fetchone()['count']
|
||||||
|
|
||||||
|
def get_draft(self, draft_id: int) -> Optional[Dict[str, Any]]:
|
||||||
|
"""获取单个草稿"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("SELECT * FROM drafts WHERE id = ?", (draft_id,))
|
||||||
|
row = cursor.fetchone()
|
||||||
|
return dict(row) if row else None
|
||||||
|
|
||||||
|
def delete_draft(self, draft_id: int) -> bool:
|
||||||
|
"""删除草稿"""
|
||||||
|
with self.get_conn() as conn:
|
||||||
|
cursor = conn.cursor()
|
||||||
|
cursor.execute("DELETE FROM drafts WHERE id = ?", (draft_id,))
|
||||||
|
conn.commit()
|
||||||
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
def draft_to_item(self, draft_id: int) -> Optional[int]:
|
||||||
|
"""将草稿转为正式条目"""
|
||||||
|
draft = self.get_draft(draft_id)
|
||||||
|
if not draft:
|
||||||
|
return None
|
||||||
|
|
||||||
|
# 创建条目
|
||||||
|
tags_list = draft['tags'].split(',') if draft['tags'] else []
|
||||||
|
tags_list = [t.strip() for t in tags_list if t.strip()]
|
||||||
|
|
||||||
|
item_id = self.create_item(
|
||||||
|
type=draft['type'],
|
||||||
|
title=draft['title'],
|
||||||
|
content=draft['content'],
|
||||||
|
url=draft['url'],
|
||||||
|
source=draft['source'],
|
||||||
|
status=draft['status'],
|
||||||
|
priority=draft['priority'],
|
||||||
|
due_date=draft['due_date'],
|
||||||
|
note=draft['note'],
|
||||||
|
tags=tags_list,
|
||||||
|
is_starred=draft['is_starred']
|
||||||
|
)
|
||||||
|
|
||||||
|
# 删除草稿
|
||||||
|
if item_id:
|
||||||
|
self.delete_draft(draft_id)
|
||||||
|
|
||||||
|
return item_id
|
||||||
|
|
||||||
# ============ Tag 操作 ============
|
# ============ Tag 操作 ============
|
||||||
|
|
||||||
def create_tag(self, name: str, color: str = "#3498db") -> int:
|
def create_tag(self, name: str, color: str = "#3498db") -> int:
|
||||||
|
|||||||
Reference in New Issue
Block a user