Compare commits

..

No commits in common. "32621bcc87ea665e1ab9972f1f7aa901c3770b32" and "169d9749af2cd48d6bb4454d196265e041d1b9fc" have entirely different histories.

5 changed files with 91 additions and 144 deletions

View File

@ -12,6 +12,8 @@ except ImportError:
from PySide2 import QtCore, QtWidgets from PySide2 import QtCore, QtWidgets
PYSIDE_6 = False PYSIDE_6 = False
# noinspection PyUnresolvedReferences
import fime.icons
from fime.data import Tasks, Log, Data, LogCommentsData from fime.data import Tasks, Log, Data, LogCommentsData
from fime.exceptions import FimeException from fime.exceptions import FimeException
from fime.import_task import ImportTask from fime.import_task import ImportTask

View File

@ -1,13 +1,8 @@
import enum
try: try:
from PySide6 import QtCore, QtGui, QtWidgets from PySide6 import QtCore, QtGui, QtWidgets
except ImportError: except ImportError:
from PySide2 import QtCore, QtGui, QtWidgets from PySide2 import QtCore, QtGui, QtWidgets
# noinspection PyUnresolvedReferences
import fime.icons
def get_screen_height(qobject): def get_screen_height(qobject):
if hasattr(qobject, "screen"): if hasattr(qobject, "screen"):
@ -16,7 +11,6 @@ def get_screen_height(qobject):
print("unable to detect screen height falling back to default value of 1080") print("unable to detect screen height falling back to default value of 1080")
return 1080 return 1080
def get_screen_width(qobject): def get_screen_width(qobject):
if hasattr(qobject, "screen"): if hasattr(qobject, "screen"):
return qobject.screen().size().width() return qobject.screen().size().width()
@ -43,9 +37,3 @@ class EditStartedDetector(QtWidgets.QStyledItemDelegate):
editor.editingFinished.connect(self.editFinished) editor.editingFinished.connect(self.editFinished)
self.editStarted.emit() self.editStarted.emit()
return editor return editor
class Status(enum.Enum):
PROGRESS = enum.auto()
OK = enum.auto()
ERROR = enum.auto()

View File

@ -1,11 +1,12 @@
import enum
from datetime import datetime
from functools import reduce, partial from functools import reduce, partial
from typing import Optional, List, Tuple from typing import Optional, List, Tuple
from fime.data import Worklog from fime.data import Worklog
from fime.progressindicator import ProgressIndicator from fime.progressindicator import ProgressIndicator
from fime.task_completer import TaskCompleter from fime.task_completer import TaskCompleter
from fime.util import get_icon, EditStartedDetector, Status from fime.util import get_icon, get_screen_height, EditStartedDetector
from fime.worklog_rest import WorklogRest
try: try:
from PySide6 import QtCore, QtGui, QtWidgets from PySide6 import QtCore, QtGui, QtWidgets
@ -13,6 +14,8 @@ except ImportError:
from PySide2 import QtCore, QtGui, QtWidgets from PySide2 import QtCore, QtGui, QtWidgets
from fime.util import get_screen_width from fime.util import get_screen_width
# noinspection PyUnresolvedReferences
import fime.icons
class WorklogDialog(QtWidgets.QDialog): class WorklogDialog(QtWidgets.QDialog):
@ -80,7 +83,6 @@ class WorklogDialog(QtWidgets.QDialog):
completer.setFilterMode(QtCore.Qt.MatchFlag.MatchContains) completer.setFilterMode(QtCore.Qt.MatchFlag.MatchContains)
completer.setCaseSensitivity(QtCore.Qt.CaseInsensitive) completer.setCaseSensitivity(QtCore.Qt.CaseInsensitive)
editor.setCompleter(completer) editor.setCompleter(completer)
editor.textChanged.connect(completer.update_picker)
return editor return editor
@QtCore.Slot() @QtCore.Slot()
@ -92,20 +94,23 @@ class WorklogDialog(QtWidgets.QDialog):
if not self.return_: if not self.return_:
self.edit_finished_row.emit(row) self.edit_finished_row.emit(row)
class Status(enum.Enum):
PROGRESS = enum.auto()
OK = enum.auto()
ERROR = enum.auto()
def __init__(self, config, parent, *args, **kwargs): def __init__(self, config, parent, *args, **kwargs):
super().__init__(parent, *args, **kwargs) super().__init__(parent, *args, **kwargs)
self.rest = WorklogRest(config) self.setWindowTitle("Worklog")
self._changing_items = False self._changing_items = False
self._worklog: Optional[Worklog] = None self._worklog: Optional[Worklog] = None
self._worklog_data: List[List[str]] = [] self._worklog_data: List[List[str]] = []
self._statuses: List[Tuple[Status, str]] = [] self._status: List[Tuple[WorklogDialog.Status, str]] = []
self.row_height = None self.row_height = None
self._focussed = False self._focussed = False
self.setWindowTitle("Worklog")
self.tableWidget = WorklogDialog.TabTable(self) self.tableWidget = WorklogDialog.TabTable(self)
self.tableWidget.setColumnCount(3) self.tableWidget.setColumnCount(3)
self.tableWidget.setHorizontalHeaderLabels(["Task", "Comment", "Duration"]) self.tableWidget.setHorizontalHeaderLabels(["Task", "Comment", "Duration"])
@ -166,36 +171,18 @@ class WorklogDialog(QtWidgets.QDialog):
layout.addLayout(alayout) layout.addLayout(alayout)
self.setLayout(layout) self.setLayout(layout)
self.update_timer = QtCore.QTimer(self)
self.update_timer.setInterval(500)
self.update_timer.timeout.connect(self.update_statuses)
def set_data(self, worklog: Worklog): def set_data(self, worklog: Worklog):
self._worklog = worklog self._worklog = worklog
self.update_all()
def update_all(self):
self.refresh_table() self.refresh_table()
self.update_title() self.update_title()
self.update_prev_next() self.update_prev_next()
self.update_statuses()
self.update_timer.start()
def update_statuses(self): def update_status(self, row, new_status, status_text="Could not find specified issue"):
issue_keys = [] self._status[row] = (new_status, status_text)
for row in self._worklog_data[:-1]:
issue_keys.append(row[0].split()[0])
old_statuses = self._statuses
self._statuses = self.rest.get_issues_state(issue_keys)
for row, status in enumerate(self._statuses):
if len(old_statuses) != len(self._statuses) or old_statuses[row][0] != status[0]:
self.update_status_view(row) self.update_status_view(row)
all_ok = reduce(lambda acc, it: acc and it[0] is Status.OK, self._statuses, True) all_ok = reduce(lambda acc, it: acc and it[0] is WorklogDialog.Status.OK, self._status, True)
if all_ok: if all_ok:
self.upload_button.setEnabled(True) self.upload_button.setEnabled(True)
all_done = reduce(lambda acc, it: acc and it[0] is not Status.PROGRESS, self._statuses, True)
if all_done:
self.update_timer.stop()
def save(self): def save(self):
self._worklog.worklog = self._worklog_data self._worklog.worklog = self._worklog_data
@ -205,6 +192,8 @@ class WorklogDialog(QtWidgets.QDialog):
def refresh_table(self): def refresh_table(self):
self._worklog_data = self._worklog.worklog self._worklog_data = self._worklog.worklog
for i, _ in enumerate(self._worklog_data[:-1]):
self._status.append((WorklogDialog.Status.PROGRESS, "Fetching"))
self._changing_items = True self._changing_items = True
if not self.row_height: if not self.row_height:
self.tableWidget.setRowCount(1) self.tableWidget.setRowCount(1)
@ -223,6 +212,7 @@ class WorklogDialog(QtWidgets.QDialog):
self.tableWidget.setItem(row, 1, item1) self.tableWidget.setItem(row, 1, item1)
item1.setFlags(item1.flags() & QtCore.Qt.ItemIsEnabled) item1.setFlags(item1.flags() & QtCore.Qt.ItemIsEnabled)
else: else:
self.update_status_view(row)
text_edit = WorklogDialog.TableTextEdit(self._worklog_data[row][1], row, self) text_edit = WorklogDialog.TableTextEdit(self._worklog_data[row][1], row, self)
text_edit.textChanged.connect(self.on_resize) text_edit.textChanged.connect(self.on_resize)
text_edit.setFrameStyle(QtWidgets.QFrame.NoFrame) text_edit.setFrameStyle(QtWidgets.QFrame.NoFrame)
@ -249,19 +239,19 @@ class WorklogDialog(QtWidgets.QDialog):
self.move(x, y) self.move(x, y)
def update_status_view(self, row): def update_status_view(self, row):
if self._statuses[row][0] is Status.PROGRESS: if self._status[row][0] is WorklogDialog.Status.PROGRESS:
icon = ProgressIndicator(self) icon = ProgressIndicator(self)
icon.setMaximumSize(QtCore.QSize(self.row_height * 0.75, self.row_height * 0.75)) icon.setMaximumSize(QtCore.QSize(self.row_height * 0.75, self.row_height * 0.75))
icon.setAnimationDelay(70) icon.setAnimationDelay(70)
icon.startAnimation() icon.startAnimation()
else: else:
if self._statuses[row][0] is Status.OK: if self._status[row][0] is WorklogDialog.Status.OK:
item_name = "dialog-ok" item_name = "dialog-ok"
elif self._statuses[row][0] is Status.ERROR: elif self._status[row][0] is WorklogDialog.Status.ERROR:
item_name = "edit-delete-remove" item_name = "edit-delete-remove"
icon = QtWidgets.QLabel() icon = QtWidgets.QLabel()
icon.setPixmap(get_icon(item_name).pixmap(self.row_height * 0.75, self.row_height * 0.75)) icon.setPixmap(get_icon(item_name).pixmap(self.row_height * 0.75, self.row_height * 0.75))
icon.setToolTip(self._statuses[row][1]) icon.setToolTip(self._status[row][1])
layout = QtWidgets.QHBoxLayout() layout = QtWidgets.QHBoxLayout()
layout.addWidget(icon, alignment=QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter) layout.addWidget(icon, alignment=QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
layout.setContentsMargins(self.row_height * 0.1, self.row_height * 0.1, layout.setContentsMargins(self.row_height * 0.1, self.row_height * 0.1,
@ -296,15 +286,21 @@ class WorklogDialog(QtWidgets.QDialog):
def previous(self): def previous(self):
self.save() self.save()
self._worklog.previous() self._worklog.previous()
self.update_title()
self._focussed = False self._focussed = False
self.update_all() self.refresh_table()
self.center_dialog()
self.update_prev_next()
@QtCore.Slot() @QtCore.Slot()
def next(self): def next(self):
self.save() self.save()
self._worklog.next() self._worklog.next()
self.update_title()
self._focussed = False self._focussed = False
self.update_all() self.refresh_table()
self.center_dialog()
self.update_prev_next()
@QtCore.Slot() @QtCore.Slot()
def cell_changed(self, row, _): def cell_changed(self, row, _):
@ -312,7 +308,8 @@ class WorklogDialog(QtWidgets.QDialog):
return return
self._worklog_data[row][0] = self.tableWidget.item(row, 0).text() self._worklog_data[row][0] = self.tableWidget.item(row, 0).text()
self.save() self.save()
self.update_all() self.update_status(row, WorklogDialog.Status.PROGRESS, "Fetching")
self.refresh_table()
@QtCore.Slot() @QtCore.Slot()
def text_edit_changed(self, text, row): def text_edit_changed(self, text, row):

56
src/fime/worklog_.py Normal file
View File

@ -0,0 +1,56 @@
import os
from datetime import date, datetime
import requests
from fime.config import Config
class Worklog:
def __init__(self, config: Config):
self.config = config
self.user_url = os.path.join(config.jira_url, "rest/api/2/myself")
self.issue_url = os.path.join(config.jira_url, "rest/api/2/issue/{}")
self.worklog_url = os.path.join(config.jira_url, "rest/api/2/issue/{}/worklog")
self.worklog_update_url = os.path.join(config.jira_url, "rest/api/2/issue/{issue}/worklog/{worklog}")
self.user = self.get_user()
def get_user(self):
resp = requests.get(self.user_url,
headers={
"Authorization": f"Bearer {self.config.jira_token}",
"Accept": "application/json",
},
)
user = resp.json()["key"]
print(user)
return user
def get(self, issue_key: str, date: date):
resp = requests.get(self.issue_url.format(issue_key),
headers={
"Authorization": f"Bearer {self.config.jira_token}",
"Accept": "application/json",
},
)
if resp.status_code != 200:
raise RuntimeError("issue does not exist")
resp = requests.get(self.worklog_url.format(issue_key),
headers={
"Authorization": f"Bearer {self.config.jira_token}",
"Accept": "application/json",
},
)
worklogs = resp.json()["worklogs"]
found = False
for log in worklogs:
if log["author"]["key"] == self.user \
and datetime.strptime(log["started"], "%Y-%m-%dT%H:%M:%S.%f%z").date() == date:
print(log["id"])
print(log["comment"])
found = True
break
print(found)

View File

@ -1,96 +0,0 @@
import os
import traceback
from concurrent.futures import Future
from datetime import date, datetime
from functools import partial
from threading import Lock
from typing import List, Dict, Tuple
import requests
from requests_futures.sessions import FuturesSession
from fime.config import Config
from fime.exceptions import FimeException
from fime.util import Status
class WorklogRest:
def __init__(self, config: Config):
self.config = config
self.user_url = os.path.join(config.jira_url, "rest/api/2/myself")
self.issue_url = os.path.join(config.jira_url, "rest/api/2/issue/{}")
self.worklog_url = os.path.join(config.jira_url, "rest/api/2/issue/{}/worklog")
self.worklog_update_url = os.path.join(config.jira_url, "rest/api/2/issue/{issue}/worklog/{worklog}")
self.session = FuturesSession()
self.user = self._req_user()
self._issue_state: Dict[str, Tuple[Status, str]] = {}
self._issue_state_lock = Lock()
def _req_user(self):
return self.session.get(
self.user_url,
headers={
"Authorization": f"Bearer {self.config.jira_token}",
"Accept": "application/json",
},
)
def _get_user(self):
try:
return self.user.result().json()["key"]
except Exception:
raise FimeException("Could not get user key:\n" + traceback.format_exc())
def get_issues_state(self, issue_keys: List[str]):
ret = []
with self._issue_state_lock:
for issue_key in issue_keys:
if issue_key not in self._issue_state:
self._issue_state[issue_key] = (Status.PROGRESS, "Fetching")
self._req_issue(issue_key)
ret.append(self._issue_state[issue_key])
return ret
def _req_issue(self, issue_key: str):
future = self.session.get(self.issue_url.format(issue_key),
headers={
"Authorization": f"Bearer {self.config.jira_token}",
"Accept": "application/json",
},
)
future.add_done_callback(partial(self._resp_issue, issue_key))
def _resp_issue(self, issue_key: str, future: Future):
resp: requests.Response = future.result()
with self._issue_state_lock:
if resp.status_code == 200:
hint = f'{issue_key} {resp.json()["fields"]["summary"]}'
self._issue_state[issue_key] = (Status.OK, hint)
else:
self._issue_state[issue_key] = (Status.ERROR, "Could not find specified issue")
def get(self, issue_key: str, date: date):
resp = requests.get(self.issue_url.format(issue_key),
headers={
"Authorization": f"Bearer {self.config.jira_token}",
"Accept": "application/json",
},
)
if resp.status_code != 200:
raise RuntimeError("issue does not exist")
resp = requests.get(self.worklog_url.format(issue_key),
headers={
"Authorization": f"Bearer {self.config.jira_token}",
"Accept": "application/json",
},
)
worklogs = resp.json()["worklogs"]
found = False
for log in worklogs:
if log["author"]["key"] == self.user \
and datetime.strptime(log["started"], "%Y-%m-%dT%H:%M:%S.%f%z").date() == date:
print(log["id"])
print(log["comment"])
found = True
break
print(found)