mirror of https://github.com/gmbrax/Pilgrim.git
Compare commits
No commits in common. "447dffd7df210cff4473476ae9124e62b79d694d" and "4aca171ce316d65b4810b0d06c3e8e54aead80d1" have entirely different histories.
447dffd7df
...
4aca171ce3
|
|
@ -1,7 +1,9 @@
|
||||||
from sqlalchemy import create_engine
|
from sqlalchemy import create_engine
|
||||||
from sqlalchemy.orm import declarative_base
|
from sqlalchemy.orm import declarative_base
|
||||||
from sqlalchemy.orm import sessionmaker
|
from sqlalchemy.orm import sessionmaker
|
||||||
|
from pathlib import Path
|
||||||
import os
|
import os
|
||||||
|
import shutil
|
||||||
|
|
||||||
from pilgrim.utils import ConfigManager
|
from pilgrim.utils import ConfigManager
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,34 +0,0 @@
|
||||||
import zipfile
|
|
||||||
|
|
||||||
|
|
||||||
from pilgrim.utils.directory_manager import DirectoryManager
|
|
||||||
|
|
||||||
|
|
||||||
class BackupService:
|
|
||||||
def __init__(self, session):
|
|
||||||
self.session = session
|
|
||||||
|
|
||||||
def create_backup(self):
|
|
||||||
db_path = DirectoryManager.get_database_path()
|
|
||||||
if not db_path.exists():
|
|
||||||
raise FileNotFoundError("No Database Found")
|
|
||||||
|
|
||||||
with self.session.connection() as conn:
|
|
||||||
raw_conn = conn.connection
|
|
||||||
dump = "\n".join(line for line in raw_conn.iterdump())
|
|
||||||
|
|
||||||
filename = DirectoryManager.get_config_directory() / "backup.zip"
|
|
||||||
diaries_root_path = DirectoryManager.get_diaries_root()
|
|
||||||
|
|
||||||
try:
|
|
||||||
with zipfile.ZipFile(filename, "w",zipfile.ZIP_DEFLATED) as zipf:
|
|
||||||
zipf.writestr("database.sql", dump)
|
|
||||||
if diaries_root_path.exists():
|
|
||||||
for file_path in diaries_root_path.rglob('*'):
|
|
||||||
if file_path.is_file():
|
|
||||||
arcname = file_path.relative_to(diaries_root_path.parent)
|
|
||||||
zipf.write(file_path, arcname=arcname)
|
|
||||||
return True, filename
|
|
||||||
except Exception as e:
|
|
||||||
return False, str(e)
|
|
||||||
|
|
||||||
|
|
@ -1,6 +1,7 @@
|
||||||
from pilgrim.service.mocks.entry_service_mock import EntryServiceMock
|
from pilgrim.service.mocks.entry_service_mock import EntryServiceMock
|
||||||
from pilgrim.service.mocks.photo_service_mock import PhotoServiceMock
|
from pilgrim.service.mocks.photo_service_mock import PhotoServiceMock
|
||||||
from pilgrim.service.mocks.travel_diary_service_mock import TravelDiaryServiceMock
|
from pilgrim.service.mocks.travel_diary_service_mock import TravelDiaryServiceMock
|
||||||
|
from pilgrim.service.photo_service import PhotoService
|
||||||
from pilgrim.service.servicemanager import ServiceManager
|
from pilgrim.service.servicemanager import ServiceManager
|
||||||
|
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -3,6 +3,7 @@ import re
|
||||||
import shutil
|
import shutil
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
|
from pilgrim.models.entry import Entry
|
||||||
from pilgrim.utils import DirectoryManager
|
from pilgrim.utils import DirectoryManager
|
||||||
from sqlalchemy.exc import IntegrityError
|
from sqlalchemy.exc import IntegrityError
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,19 +1,19 @@
|
||||||
from typing import Optional, Tuple
|
from typing import Optional, Tuple
|
||||||
|
import asyncio
|
||||||
|
|
||||||
from textual.app import ComposeResult
|
from textual.app import ComposeResult
|
||||||
from textual.screen import Screen
|
from textual.screen import Screen
|
||||||
from textual.widgets import Header, Footer, Static, OptionList, Button
|
from textual.widgets import Header, Footer, Label, Static, OptionList, Button
|
||||||
from textual.binding import Binding
|
from textual.binding import Binding
|
||||||
from textual.containers import Container, Horizontal
|
from textual.containers import Vertical, Container, Horizontal
|
||||||
|
|
||||||
|
from pilgrim.models.travel_diary import TravelDiary
|
||||||
from pilgrim.ui.screens.about_screen import AboutScreen
|
from pilgrim.ui.screens.about_screen import AboutScreen
|
||||||
from pilgrim.ui.screens.diary_settings_screen import SettingsScreen
|
from pilgrim.ui.screens.diary_settings_screen import SettingsScreen
|
||||||
from pilgrim.ui.screens.edit_diary_modal import EditDiaryModal
|
from pilgrim.ui.screens.edit_diary_modal import EditDiaryModal
|
||||||
from pilgrim.ui.screens.new_diary_modal import NewDiaryModal
|
from pilgrim.ui.screens.new_diary_modal import NewDiaryModal
|
||||||
from pilgrim.ui.screens.edit_entry_screen import EditEntryScreen
|
from pilgrim.ui.screens.edit_entry_screen import EditEntryScreen
|
||||||
|
|
||||||
from pilgrim.service.backup_service import BackupService
|
|
||||||
|
|
||||||
|
|
||||||
class DiaryListScreen(Screen):
|
class DiaryListScreen(Screen):
|
||||||
TITLE = "Pilgrim - Main"
|
TITLE = "Pilgrim - Main"
|
||||||
|
|
@ -211,11 +211,11 @@ class DiaryListScreen(Screen):
|
||||||
def _on_new_diary_submitted(self, result):
|
def _on_new_diary_submitted(self, result):
|
||||||
"""Callback after diary creation"""
|
"""Callback after diary creation"""
|
||||||
if result: # Se result não é string vazia, o diário foi criado
|
if result: # Se result não é string vazia, o diário foi criado
|
||||||
self.notify("Returning to diary list...")
|
self.notify(f"Returning to diary list...")
|
||||||
# Atualiza a lista de diários
|
# Atualiza a lista de diários
|
||||||
self.refresh_diaries()
|
self.refresh_diaries()
|
||||||
else:
|
else:
|
||||||
self.notify("Creation canceled...")
|
self.notify(f"Creation canceled...")
|
||||||
|
|
||||||
def _on_screen_resume(self) -> None:
|
def _on_screen_resume(self) -> None:
|
||||||
super()._on_screen_resume()
|
super()._on_screen_resume()
|
||||||
|
|
@ -298,18 +298,3 @@ class DiaryListScreen(Screen):
|
||||||
self.notify("Invalid diary ID")
|
self.notify("Invalid diary ID")
|
||||||
else:
|
else:
|
||||||
self.notify("Select a diary to open the settings")
|
self.notify("Select a diary to open the settings")
|
||||||
|
|
||||||
|
|
||||||
def action_backup(self):
|
|
||||||
session = self.app.service_manager.get_session()
|
|
||||||
if session:
|
|
||||||
backup_service = BackupService(session)
|
|
||||||
result_operation, result_data = backup_service.create_backup()
|
|
||||||
if result_operation:
|
|
||||||
self.notify(f"Backup result: {result_data}")
|
|
||||||
else:
|
|
||||||
self.notify(f"Error performing backup: {result_data}")
|
|
||||||
else:
|
|
||||||
self.notify("Error: Session not found",severity="error")
|
|
||||||
self.app.exit()
|
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,13 @@
|
||||||
|
|
||||||
from textual.widgets import Static
|
from textual.widgets import Static
|
||||||
from textual.containers import Container
|
from textual.containers import Container
|
||||||
from textual.widgets import Header, Footer, Label, Button,Checkbox
|
from textual.widgets import Header, Footer, Label, Button,Checkbox,Input
|
||||||
from textual.screen import Screen
|
from textual.screen import Screen
|
||||||
from textual.reactive import reactive
|
from textual.reactive import reactive
|
||||||
from textual.binding import Binding
|
from textual.binding import Binding
|
||||||
from textual import on
|
from textual import on
|
||||||
|
|
||||||
|
from pilgrim.models.travel_diary import TravelDiary
|
||||||
from pilgrim.ui.screens.modals.delete_all_entries_from_diary_modal import DeleteAllEntriesModal
|
from pilgrim.ui.screens.modals.delete_all_entries_from_diary_modal import DeleteAllEntriesModal
|
||||||
from pilgrim.ui.screens.modals.delete_all_photos_from_diary_modal import DeleteAllPhotosModal
|
from pilgrim.ui.screens.modals.delete_all_photos_from_diary_modal import DeleteAllPhotosModal
|
||||||
from pilgrim.ui.screens.modals.delete_diary_modal import DeleteDiaryModal
|
from pilgrim.ui.screens.modals.delete_diary_modal import DeleteDiaryModal
|
||||||
|
|
|
||||||
|
|
@ -5,9 +5,11 @@ from typing import Optional, List
|
||||||
|
|
||||||
from pilgrim.models.entry import Entry
|
from pilgrim.models.entry import Entry
|
||||||
from pilgrim.models.photo import Photo
|
from pilgrim.models.photo import Photo
|
||||||
|
from pilgrim.models.travel_diary import TravelDiary
|
||||||
from pilgrim.ui.screens.modals.add_photo_modal import AddPhotoModal
|
from pilgrim.ui.screens.modals.add_photo_modal import AddPhotoModal
|
||||||
from pilgrim.ui.screens.modals.confirm_delete_modal import ConfirmDeleteModal
|
from pilgrim.ui.screens.modals.confirm_delete_modal import ConfirmDeleteModal
|
||||||
from pilgrim.ui.screens.modals.edit_photo_modal import EditPhotoModal
|
from pilgrim.ui.screens.modals.edit_photo_modal import EditPhotoModal
|
||||||
|
from pilgrim.ui.screens.modals.file_picker_modal import FilePickerModal
|
||||||
from pilgrim.ui.screens.rename_entry_modal import RenameEntryModal
|
from pilgrim.ui.screens.rename_entry_modal import RenameEntryModal
|
||||||
from textual.app import ComposeResult
|
from textual.app import ComposeResult
|
||||||
from textual.binding import Binding
|
from textual.binding import Binding
|
||||||
|
|
@ -187,7 +189,7 @@ class EditEntryScreen(Screen):
|
||||||
"""Ensures the diary info widget is always updated with current diary name"""
|
"""Ensures the diary info widget is always updated with current diary name"""
|
||||||
try:
|
try:
|
||||||
self.diary_info.update(f"Diary: {self.diary_name}")
|
self.diary_info.update(f"Diary: {self.diary_name}")
|
||||||
except Exception:
|
except Exception as e:
|
||||||
self.diary_info.update(f"Diary: {self.diary_id}")
|
self.diary_info.update(f"Diary: {self.diary_id}")
|
||||||
|
|
||||||
def refresh_entries(self):
|
def refresh_entries(self):
|
||||||
|
|
@ -425,7 +427,7 @@ class EditEntryScreen(Screen):
|
||||||
photo_details += f"🔗 {photo_hash}\n"
|
photo_details += f"🔗 {photo_hash}\n"
|
||||||
photo_details += f"📅 {selected_photo.addition_date}\n"
|
photo_details += f"📅 {selected_photo.addition_date}\n"
|
||||||
photo_details += f"💬 {selected_photo.caption or 'No caption'}\n"
|
photo_details += f"💬 {selected_photo.caption or 'No caption'}\n"
|
||||||
photo_details += "[b]Reference formats:[/b]\n"
|
photo_details += f"[b]Reference formats:[/b]\n"
|
||||||
photo_details += f"\\[\\[photo::{photo_hash}\\]\\]"
|
photo_details += f"\\[\\[photo::{photo_hash}\\]\\]"
|
||||||
|
|
||||||
self.photo_info.update(photo_details)
|
self.photo_info.update(photo_details)
|
||||||
|
|
@ -743,8 +745,8 @@ class EditEntryScreen(Screen):
|
||||||
if selected_photo.caption:
|
if selected_photo.caption:
|
||||||
photo_details += f"Caption: {selected_photo.caption}\n"
|
photo_details += f"Caption: {selected_photo.caption}\n"
|
||||||
else:
|
else:
|
||||||
photo_details += "Caption: No Caption\n"
|
photo_details += f"Caption: No Caption\n"
|
||||||
photo_details += "[b]Reference formats:[/b]\n"
|
photo_details += f"[b]Reference formats:[/b]\n"
|
||||||
photo_details += f"\\[\\[photo::{photo_hash}]]"
|
photo_details += f"\\[\\[photo::{photo_hash}]]"
|
||||||
|
|
||||||
self.photo_info.update(photo_details)
|
self.photo_info.update(photo_details)
|
||||||
|
|
|
||||||
|
|
@ -1,9 +1,11 @@
|
||||||
|
import os
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from textual.app import ComposeResult
|
from textual.app import ComposeResult
|
||||||
from textual.screen import Screen
|
from textual.screen import Screen
|
||||||
from textual.widgets import Static, Input, Button
|
from textual.widgets import Static, Input, Button
|
||||||
from textual.containers import Horizontal, Container
|
from textual.containers import Horizontal, Container
|
||||||
from .file_picker_modal import FilePickerModal
|
from .file_picker_modal import FilePickerModal
|
||||||
|
import hashlib
|
||||||
|
|
||||||
class AddPhotoModal(Screen):
|
class AddPhotoModal(Screen):
|
||||||
"""Modal for adding a new photo"""
|
"""Modal for adding a new photo"""
|
||||||
|
|
|
||||||
|
|
@ -37,11 +37,6 @@ class UIApp(App):
|
||||||
"Open About Pilgrim",
|
"Open About Pilgrim",
|
||||||
screen.action_about_cmd
|
screen.action_about_cmd
|
||||||
)
|
)
|
||||||
yield SystemCommand(
|
|
||||||
"Backup Database",
|
|
||||||
"Backup the Database",
|
|
||||||
screen.action_backup
|
|
||||||
)
|
|
||||||
|
|
||||||
elif isinstance(screen, AboutScreen):
|
elif isinstance(screen, AboutScreen):
|
||||||
yield SystemCommand(
|
yield SystemCommand(
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
import os.path
|
import os.path
|
||||||
|
from os import PathLike
|
||||||
from threading import Lock
|
from threading import Lock
|
||||||
|
|
||||||
import tomli
|
import tomli
|
||||||
|
|
|
||||||
|
|
@ -1,8 +1,14 @@
|
||||||
|
from re import search
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
|
from unittest.mock import Mock
|
||||||
|
from sqlalchemy import create_engine
|
||||||
from sqlalchemy.exc import IntegrityError
|
from sqlalchemy.exc import IntegrityError
|
||||||
|
from sqlalchemy.orm import sessionmaker
|
||||||
|
|
||||||
|
from pilgrim.database import Base
|
||||||
|
from pilgrim.models.travel_diary import TravelDiary
|
||||||
from pilgrim.models.entry import Entry
|
from pilgrim.models.entry import Entry
|
||||||
from pilgrim.models.photo import Photo
|
from pilgrim.models.photo import Photo
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,4 @@
|
||||||
|
import pytest
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
from pilgrim import TravelDiary
|
from pilgrim import TravelDiary
|
||||||
|
|
|
||||||
|
|
@ -4,6 +4,7 @@ from pathlib import Path
|
||||||
from unittest.mock import patch
|
from unittest.mock import patch
|
||||||
|
|
||||||
from pilgrim.utils.config_manager import ConfigManager, SingletonMeta
|
from pilgrim.utils.config_manager import ConfigManager, SingletonMeta
|
||||||
|
from pilgrim.utils.directory_manager import DirectoryManager
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def clean_singleton():
|
def clean_singleton():
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue