awork_tasks.py 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. import datetime
  2. import json
  3. from pathlib import Path
  4. import requests
  5. import typer
  6. from credentials import Client
  7. from model import AworkProjekt, Comment, User
  8. from sqlalchemy import create_engine, text
  9. from sqlalchemy.orm import Session
  10. app = typer.Typer()
  11. DSN = "mysql+pymysql://gaps:Gcbs12ma@192.168.2.41/tasks"
  12. awork_api_url = "https://api.awork.com/api/v1"
  13. token_expires_in = 24 * 60 * 60
  14. header = {"Authorization": ""}
  15. token_json_file = Path(__file__).parent.joinpath("token.json")
  16. access_token_file = Path(__file__).parent.joinpath("access_token.txt")
  17. def main():
  18. # bearer_token = login()
  19. bearer_token = access_token_file.read_text()
  20. header["Authorization"] = f"Bearer {bearer_token}"
  21. # users = get_users()
  22. # companies = get_companies()
  23. # import_projects()
  24. # create_task(73849)
  25. def get_companies():
  26. res = requests.get(awork_api_url + "/companies", headers=header)
  27. data = res.json()
  28. json.dump(data, Path(__file__).parent.joinpath("companies.json").open("w"), indent=2)
  29. return data
  30. def get_users():
  31. res = requests.get(awork_api_url + "/users", headers=header)
  32. data = res.json()
  33. json.dump(data, Path(__file__).parent.joinpath("users.json").open("w"), indent=2)
  34. return data
  35. @app.command()
  36. def import_projects():
  37. projects = get_projects()
  38. p = convert_projects(projects)
  39. with Session(create_engine(DSN)) as db_session:
  40. db_session.execute(text("TRUNCATE TABLE awork_projekte"))
  41. db_session.add_all(p)
  42. db_session.commit()
  43. def get_projects():
  44. res = requests.get(awork_api_url + "/projects?pageSize=1000", headers=header)
  45. data = res.json()
  46. json.dump(data, Path(__file__).parent.joinpath("projects.json").open("w"), indent=2)
  47. return data
  48. def convert_projects(projects):
  49. res = []
  50. for p in projects:
  51. if "companyId" not in p:
  52. continue
  53. p_allgemein = 1 if "Allgemein" in p["name"] else 0
  54. p_typ = p["projectType"]["name"] if "projectType" in p else "Unbekannt"
  55. entry = AworkProjekt(
  56. **{
  57. "awork_project_id": p["id"],
  58. "awork_company_id": p["companyId"],
  59. "projekt_name": p["name"],
  60. "projekt_status": p["projectStatus"]["name"],
  61. "projekt_typ": p_typ,
  62. "projekt_allgemein": p_allgemein,
  63. "kunde_name": p["company"]["name"],
  64. "tasks_count": p["tasksCount"],
  65. "tracked_duration": p["trackedDuration"],
  66. }
  67. )
  68. res.append(entry)
  69. return res
  70. @app.command()
  71. def create_task(comment_id: int):
  72. with Session(create_engine(DSN)) as db_session:
  73. comment = db_session.get(Comment, comment_id)
  74. if comment.awork_task_id:
  75. return comment.awork_task_id
  76. datum = comment.datum.strftime("%d.%m.%Y")
  77. link = f"http://gc-server1/fehlerbericht/#/report/{comment.kunde}/{comment.datum}/{comment.start}"
  78. res = requests.get(awork_api_url + "/projects/" + comment.awork_project_id + "/taskstatuses", headers=header)
  79. data = res.json()
  80. task_status = [s["id"] for s in data if s["type"] == "progress"]
  81. if len(task_status) == 0:
  82. return 0
  83. task_status_id = task_status[0]
  84. task = {
  85. "name": f"{comment.kunde} - Fehlerbericht vom {datum} - {comment.fehler} Fehler",
  86. "description": (
  87. f'<p><a target="_blank" rel="noopener noreferrer nofollow" href="{link}">Fehlerbericht-Portal</a></p>'
  88. + f"<p>{comment.benutzer}: {comment.kommentar}</p>"
  89. ),
  90. "isPrio": False,
  91. # "startOn": "2021-03-03T17:00:00Z",
  92. "dueOn": f"{comment.datum}T16:00:00Z",
  93. # "laneOrder": 0,
  94. "plannedDuration": 3600,
  95. # "remainingDuration": 0,
  96. "typeOfWorkId": "1854242b-8d9a-44c6-a98e-f17b89b6bed8",
  97. "taskStatusId": task_status_id,
  98. # "order": 0,
  99. # "subtaskOrder": 1,
  100. "entityId": comment.awork_project_id,
  101. "baseType": "projecttask",
  102. # "parentId": "3fa85f64-5717-4562-b3fc-2c963f66afa6",
  103. # "lists": [{"id": "94166142-b5f9-4bd0-968d-a6a3474ee705", "order": 0}],
  104. }
  105. res = requests.post(awork_api_url + "/tasks", json=task, headers=header)
  106. data = res.json()
  107. task_id = data["id"]
  108. res = requests.post(
  109. awork_api_url + f"/tasks/{task_id}/addtags", json=[{"name": "Fehlerprotokolle"}], headers=header
  110. )
  111. user1 = db_session.get(User, comment.benutzer)
  112. user_comment = {
  113. "message": comment.kommentar,
  114. "userId": user1.awork_user_id,
  115. }
  116. res = requests.post(awork_api_url + f"/tasks/{task_id}/comments", json=user_comment, headers=header)
  117. res = requests.post(
  118. awork_api_url + f"/tasks/{task_id}/setassignees", json=[user1.awork_user_id], headers=header
  119. )
  120. user2 = user1
  121. if comment.benutzer2:
  122. user2 = db_session.get(User, comment.benutzer2)
  123. res = requests.post(
  124. awork_api_url + f"/tasks/{task_id}/setassignees", json=[user2.awork_user_id], headers=header
  125. )
  126. comment.awork_task_id = task_id
  127. db_session.commit()
  128. def login():
  129. # token_file = Path(__file__).parent.joinpath("access_token.txt")
  130. timestamp = datetime.datetime.now().timestamp()
  131. if not token_json_file.exists():
  132. return token(authorize())
  133. token_dict = json.loads(token_json_file.read_text())
  134. if timestamp < token_dict["expires_at"]:
  135. return token_dict["access_token"]
  136. res = refresh(token_dict["refresh_token"])
  137. if res:
  138. return res
  139. token_json_file.unlink()
  140. return token(authorize())
  141. def refresh(refresh_token):
  142. header = {
  143. "Authorization": "Basic Base64(" + Client.authorization_base64() + ")",
  144. "Content-Type": "application/x-www-form-urlencoded",
  145. }
  146. params = {
  147. "client_id": Client.CLIENT_ID,
  148. "client_secret": Client.CLIENT_SECRET,
  149. "grant_type": "refresh_token",
  150. "redirect_uri": "http://gc-server1/fehlerbericht/",
  151. "refresh_token": refresh_token,
  152. }
  153. r = requests.post(awork_api_url + "/accounts/token", headers=header, data=params)
  154. if r.status_code == 400:
  155. return False
  156. token_dict = r.json()
  157. if "refresh_token" not in token_dict:
  158. token_dict["refresh_token"] = refresh_token
  159. write_token_file(token_dict)
  160. return token_dict["access_token"]
  161. def token(code):
  162. header = {
  163. "Authorization": "Basic Base64(" + Client.authorization_base64() + ")",
  164. "Content-Type": "application/x-www-form-urlencoded",
  165. }
  166. params = {
  167. "client_id": Client.CLIENT_ID,
  168. "client_secret": Client.CLIENT_SECRET,
  169. "grant_type": "authorization_code",
  170. "redirect_uri": "http://gc-server1/fehlerbericht/",
  171. "code": code,
  172. }
  173. r = requests.post(awork_api_url + "/accounts/token", headers=header, data=params)
  174. if r.status_code == 400:
  175. return False
  176. token_dict = r.json()
  177. write_token_file(token_dict)
  178. return token_dict["access_token"]
  179. def write_token_file(token_dict):
  180. timestamp = datetime.datetime.now().timestamp()
  181. token_dict["expires_at"] = token_dict["expires_in"] + timestamp
  182. token_json_file.write_text(json.dumps(token_dict))
  183. def authorize():
  184. params = {
  185. "client_id": Client.CLIENT_ID,
  186. "response_type": "code",
  187. "grant_type": "authorization_code",
  188. "redirect_uri": "http://gc-server1/fehlerbericht/",
  189. "state": "",
  190. "scope": "offline_access",
  191. }
  192. r = requests.get(awork_api_url + "/accounts/authorize", params=params)
  193. print(r.url)
  194. print("code: ", end="")
  195. code = input()
  196. return code
  197. if __name__ == "__main__":
  198. main()
  199. app()
  200. # create_task(74715)
  201. # import_projects()