zou 0.20.76__py3-none-any.whl → 0.20.77__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -6,12 +6,14 @@ from zou.app.services import (
6
6
  names_service,
7
7
  persons_service,
8
8
  projects_service,
9
+ shots_service,
9
10
  tasks_service,
10
11
  )
11
12
  from zou.app.stores import queue_store
13
+ from zou.app.services.template_services import generate_html_body
12
14
 
13
15
 
14
- def send_notification(person_id, subject, messages):
16
+ def send_notification(person_id, subject, messages, title=""):
15
17
  """
16
18
  Send email notification to given person. Use the job queue if it is
17
19
  activated.
@@ -21,20 +23,20 @@ def send_notification(person_id, subject, messages):
21
23
  slack_message = messages["slack_message"]
22
24
  mattermost_message = messages["mattermost_message"]
23
25
  discord_message = messages["discord_message"]
26
+ email_html_body = generate_html_body(title, email_message)
27
+
24
28
  if person["notifications_enabled"]:
25
29
  if config.ENABLE_JOB_QUEUE:
26
30
  queue_store.job_queue.enqueue(
27
31
  emails.send_email,
28
32
  args=(
29
33
  subject,
30
- email_message + get_signature(),
34
+ email_html_body,
31
35
  person["email"],
32
36
  ),
33
37
  )
34
38
  else:
35
- emails.send_email(
36
- subject, email_message + get_signature(), person["email"]
37
- )
39
+ emails.send_email(subject, email_html_body, person["email"])
38
40
 
39
41
  if person["notifications_slack_enabled"]:
40
42
  organisation = persons_service.get_organisation(sensitive=True)
@@ -152,6 +154,7 @@ _%s_
152
154
  task_url,
153
155
  task_status_name,
154
156
  )
157
+
155
158
  messages = {
156
159
  "email_message": email_message,
157
160
  "slack_message": slack_message,
@@ -260,6 +263,7 @@ def send_assignation_notification(person_id, author_id, task):
260
263
  task_name,
261
264
  task_url,
262
265
  )
266
+
263
267
  messages = {
264
268
  "email_message": email_message,
265
269
  "slack_message": slack_message,
@@ -273,20 +277,6 @@ def send_assignation_notification(person_id, author_id, task):
273
277
  return True
274
278
 
275
279
 
276
- def get_signature():
277
- """
278
- Build signature for Zou emails.
279
- """
280
- organisation = persons_service.get_organisation()
281
- return (
282
- """
283
- <p>Best,</p>
284
-
285
- <p>%s Team</p>"""
286
- % organisation["name"]
287
- )
288
-
289
-
290
280
  def get_task_descriptors(person_id, task):
291
281
  """
292
282
  Build task information needed to write notification emails: author object,
@@ -334,6 +324,8 @@ def send_reply_notification(person_id, author_id, comment, task, reply):
334
324
  if (
335
325
  person["notifications_enabled"]
336
326
  or person["notifications_slack_enabled"]
327
+ or person["notifications_mattermost_enabled"]
328
+ or person["notifications_discord_enabled"]
337
329
  ):
338
330
  tasks_service.get_task_status(task["task_status_id"])
339
331
  project = projects_service.get_project(task["project_id"])
@@ -382,3 +374,55 @@ _%s_
382
374
  }
383
375
  send_notification(person_id, subject, messages)
384
376
  return True
377
+
378
+
379
+ def send_playlist_ready_notification(person_id, author_id, playlist):
380
+ """
381
+ Send a notification email telling that a new playlist is ready to person
382
+ matching given person id.
383
+ """
384
+ person = persons_service.get_person(person_id)
385
+ author = persons_service.get_person(author_id)
386
+ project = projects_service.get_project(playlist["project_id"])
387
+ episode = None
388
+ try:
389
+ episode = shots_service.get_episode(playlist["episode_id"])
390
+ except:
391
+ pass
392
+
393
+ if (
394
+ True
395
+ or person["notifications_enabled"]
396
+ or person["notifications_slack_enabled"]
397
+ or person["notifications_mattermost_enabled"]
398
+ or person["notifications_discord_enabled"]
399
+ ):
400
+ if episode is not None:
401
+ playlist_url = f"{config.DOMAIN_PROTOCOL}://{config.DOMAIN_NAME}/productions/{playlist['project_id']}/episodes/{episode['id']}/playlists/{playlist['id']}"
402
+ else:
403
+ playlist_url = f"{config.DOMAIN_PROTOCOL}://{config.DOMAIN_NAME}/productions/{playlist['project_id']}/playlists/{playlist['id']}"
404
+
405
+ title = "Playlist Ready"
406
+ episode_segment = ""
407
+ if episode is not None:
408
+ episode_segment = f"the episode {episode['name']} of"
409
+ subject = "[Kitsu] A new playlist is ready"
410
+
411
+ email_message = f"""<p><strong>{author["full_name"]}</strong> notifies you that playlist <a href="{playlist_url}">{playlist["name"]}</a> is ready for a review under {episode_segment} the project {project["name"]}.</p>
412
+
413
+ {len(playlist["shots"])} elements are listed in the playlist.
414
+ """
415
+
416
+ slack_message = f"*{author['full_name']}* notifies you that a playlist <{playlist_url}|{playlist['name']}> is ready for a review under {episode_segment} the project {project['name']}."
417
+
418
+ discord_message = f"*{author['full_name']}* notifies you that a playlist [{playlist['name']}]({playlist_url}) is ready for a review under {episode_segment} the project {project['name']}."
419
+ messages = {
420
+ "email_message": email_message,
421
+ "slack_message": slack_message,
422
+ "mattermost_message": {
423
+ "message": slack_message,
424
+ "project_name": project["name"],
425
+ },
426
+ "discord_message": discord_message,
427
+ }
428
+ send_notification(person_id, subject, messages, title)
@@ -135,9 +135,7 @@ def get_last_news_for_project(
135
135
  )
136
136
 
137
137
  if task_status_id is not None:
138
- query = query.filter(Comment.task_status_id == task_status_id).filter(
139
- News.change == True
140
- )
138
+ query = query.filter(Comment.task_status_id == task_status_id)
141
139
 
142
140
  if task_type_id is not None:
143
141
  query = query.filter(Task.task_type_id == task_type_id)
@@ -1,8 +1,9 @@
1
1
  from sqlalchemy.exc import StatementError
2
2
 
3
- from zou.app.models.project import Project
3
+ from zou.app.models.project import Project, ProjectPersonLink
4
4
  from zou.app.models.entity import Entity
5
5
  from zou.app.models.notification import Notification
6
+ from zou.app.models.person import Person
6
7
  from zou.app.models.subscription import Subscription
7
8
  from zou.app.models.task import Task
8
9
  from zou.app.models.task_type import TaskType
@@ -15,7 +16,7 @@ from zou.app.services import (
15
16
  persons_service,
16
17
  )
17
18
  from zou.app.services.exception import PersonNotFoundException
18
- from zou.app.utils import events, fields, query as query_utils
19
+ from zou.app.utils import date_helpers, events, fields, query as query_utils
19
20
 
20
21
  from zou.app.utils import cache
21
22
 
@@ -39,6 +40,7 @@ def create_notification(
39
40
  change=False,
40
41
  type="comment",
41
42
  created_at=None,
43
+ playlist_id=None,
42
44
  ):
43
45
  """
44
46
  Create a new notification for given person and comment.
@@ -52,6 +54,7 @@ def create_notification(
52
54
  task_id=task_id,
53
55
  comment_id=comment_id,
54
56
  reply_id=reply_id,
57
+ playlist_id=playlist_id,
55
58
  type=type,
56
59
  created_at=creation_date,
57
60
  )
@@ -492,3 +495,37 @@ def get_subscriptions_for_user(project_id, entity_type_id=None):
492
495
  for subscription in subscriptions:
493
496
  subscription_map[str(subscription.task_id)] = True
494
497
  return subscription_map
498
+
499
+
500
+ def notify_clients_playlist_ready(playlist, studio_id=None):
501
+ """
502
+ Notify clients that given playlist is ready.
503
+ """
504
+
505
+ author = persons_service.get_current_user()
506
+ project_id = playlist["project_id"]
507
+ query = (
508
+ Person.query
509
+ .join(ProjectPersonLink)
510
+ .filter(Person.is_bot == False)
511
+ .filter(Person.role == "client")
512
+ .filter(ProjectPersonLink.project_id == project_id)
513
+ )
514
+
515
+ if studio_id is not None and studio_id != "":
516
+ query = query.filter(Person.studio_id == studio_id)
517
+
518
+ for client in query.all():
519
+ recipient_id = str(client.id)
520
+ author_id = author["id"]
521
+ created_at = date_helpers.get_now()
522
+ notification = create_notification(
523
+ recipient_id,
524
+ author_id=author_id,
525
+ playlist_id=playlist["id"],
526
+ type="playlist-ready",
527
+ created_at=created_at,
528
+ )
529
+ emails_service.send_playlist_ready_notification(
530
+ recipient_id, author_id, playlist
531
+ )
@@ -0,0 +1,129 @@
1
+ from zou.app.services import persons_service
2
+
3
+ notification_template_begin = """
4
+ <!DOCTYPE html>
5
+ <html>
6
+ <head>
7
+ <style>
8
+ body {
9
+ background-color: #f3f3f3;
10
+ color: #333333;
11
+ font-family: Lato, Arial, sans-serif;
12
+ line-height: 1.6;
13
+ }
14
+
15
+ a {
16
+ color: #6B6;
17
+ font-weight: bold;
18
+ text-decoration: none;
19
+ }
20
+
21
+ .email-container {
22
+ background-color: #ffffff;
23
+ border: 1px solid #dddddd;
24
+ border-radius: 20px;
25
+ margin: 30px auto;
26
+ max-width: 600px;
27
+ width: 100%;
28
+ }
29
+
30
+ .header {
31
+ border-radius: 20px 20px 0 0;
32
+ padding: 40px 20px 0 20px;
33
+ text-align: center;
34
+ }
35
+
36
+ .content {
37
+ padding: 40px;
38
+ }
39
+
40
+ .image {
41
+ height: auto;
42
+ max-width: 100%;
43
+ }
44
+
45
+ .cta {
46
+ margin: 40px 0;
47
+ text-align: center;
48
+ }
49
+
50
+ .cta a {
51
+ background-color: #00b242;
52
+ border-radius: 20px;
53
+ color: white;
54
+ font-size: 16px;
55
+ font-weight: bold;
56
+ max-with: 300px;
57
+ padding: 10px 20px;
58
+ text-align: center;
59
+ text-decoration: none;
60
+ }
61
+
62
+ .footer {
63
+ color: #666;
64
+ font-size: 12px;
65
+ margin-left: 10px;
66
+ padding: 10px;
67
+ text-align: center;
68
+
69
+ a {
70
+ color: #666;
71
+ }
72
+ }
73
+
74
+ .address {
75
+ margin-top: 20px;
76
+ line-height: 1.5;
77
+ }
78
+ </style>
79
+ </head>
80
+ <body>
81
+ <div class="email-container">
82
+ <div class="header">
83
+ <h1>
84
+ """
85
+
86
+
87
+ notification_template_end = """
88
+ </div>
89
+ <div class="footer">
90
+ <img
91
+ width="30"
92
+ src=""
93
+ />
94
+ <br>
95
+ <strong>Better, Faster, Together</strong><br><br>
96
+ </div>
97
+ </div>
98
+ </body>
99
+ </html>
100
+ """
101
+
102
+ notification_template_end_title = """
103
+ </h1>
104
+ </div>
105
+
106
+ <div class="content">
107
+ """
108
+
109
+
110
+
111
+ def get_signature():
112
+ """
113
+ Build signature for Zou emails.
114
+ """
115
+ organisation = persons_service.get_organisation()
116
+ return (
117
+ """
118
+ <p>Best,</p>
119
+
120
+ <p>%s Team</p>"""
121
+ % organisation["name"]
122
+ )
123
+
124
+
125
+ def generate_html_body(title, message):
126
+ return notification_template_begin + \
127
+ title + notification_template_end_title + \
128
+ message + get_signature () + \
129
+ notification_template_end
@@ -633,9 +633,9 @@ def get_day_offs_between_for_project(
633
633
  for day_off in days_offs:
634
634
  day_off_person_id = str(day_off.person_id)
635
635
  result[day_off_person_id].append(
636
- day_off.serialize_safe()
637
- if safe and current_user_id != day_off_person_id
638
- else day_off.serialize()
636
+ day_off.serialize()()
637
+ if safe or current_user_id == day_off_person_id
638
+ else day_off.serialize_safe()
639
639
  )
640
640
  except DataError:
641
641
  raise WrongDateFormatException
@@ -24,6 +24,7 @@ from zou.app.services import (
24
24
  notifications_service,
25
25
  names_service,
26
26
  persons_service,
27
+ playlists_service,
27
28
  projects_service,
28
29
  shots_service,
29
30
  status_automations_service,
@@ -1384,8 +1385,8 @@ def get_last_notifications(
1384
1385
  Notification.query.filter_by(person_id=current_user["id"])
1385
1386
  .order_by(Notification.created_at.desc())
1386
1387
  .join(Author, Author.id == Notification.author_id)
1387
- .join(Task, Task.id == Notification.task_id)
1388
- .join(Project, Project.id == Task.project_id)
1388
+ .outerjoin(Task, Task.id == Notification.task_id)
1389
+ .outerjoin(Project, Project.id == Task.project_id)
1389
1390
  .outerjoin(
1390
1391
  Subscription,
1391
1392
  and_(
@@ -1442,6 +1443,7 @@ def get_last_notifications(
1442
1443
  query = query.filter(Subscription.id == None)
1443
1444
 
1444
1445
  notifications = query.limit(100).all()
1446
+ print(len(notifications))
1445
1447
 
1446
1448
  for (
1447
1449
  notification,
@@ -1456,9 +1458,22 @@ def get_last_notifications(
1456
1458
  subscription_id,
1457
1459
  role,
1458
1460
  ) in notifications:
1459
- (full_entity_name, episode_id, entity_preview_file_id) = (
1460
- names_service.get_full_entity_name(task_entity_id)
1461
- )
1461
+ full_entity_name, episode_id, entity_preview_file_id = "", None, None
1462
+ playlist_id = notification.playlist_id
1463
+ playlist_name = ""
1464
+ if notification.playlist_id is None:
1465
+ (full_entity_name, episode_id, entity_preview_file_id) = (
1466
+ names_service.get_full_entity_name(task_entity_id)
1467
+ )
1468
+ else:
1469
+ playlist = playlists_service.get_playlist(
1470
+ notification.playlist_id
1471
+ )
1472
+ episode_id = playlist.get("episode_id", None)
1473
+ project = projects_service.get_project(playlist["project_id"])
1474
+ project_id = project["id"]
1475
+ project_name = project["name"]
1476
+ playlist_name = playlist["name"]
1462
1477
  preview_file_id = None
1463
1478
  mentions = []
1464
1479
  department_mentions = []
@@ -1522,6 +1537,8 @@ def get_last_notifications(
1522
1537
  "episode_id": episode_id,
1523
1538
  "entity_preview_file_id": entity_preview_file_id,
1524
1539
  "subscription_id": subscription_id,
1540
+ "playlist_id": playlist_id,
1541
+ "playlist_name": playlist_name,
1525
1542
  }
1526
1543
  )
1527
1544
  )
zou/app/utils/commands.py CHANGED
@@ -663,7 +663,7 @@ def reset_search_index():
663
663
  with app.app_context():
664
664
  print("Resetting search index.")
665
665
  index_service.reset_index()
666
- print("Search index resetted.")
666
+ print("Search index reset.")
667
667
 
668
668
 
669
669
  def search_asset(query):
@@ -0,0 +1,35 @@
1
+ """add performance indexes
2
+
3
+ Revision ID: 0c05b22194f3
4
+ Revises: 1b0ab951adca
5
+ Create Date: 2025-09-12 16:03:33.551737
6
+
7
+ """
8
+ from alembic import op
9
+ import sqlalchemy as sa
10
+ import sqlalchemy_utils
11
+
12
+
13
+ # revision identifiers, used by Alembic.
14
+ revision = '0c05b22194f3'
15
+ down_revision = '1b0ab951adca'
16
+ branch_labels = None
17
+ depends_on = None
18
+
19
+
20
+ def upgrade():
21
+ # ### commands auto generated by Alembic - please adjust! ###
22
+ with op.batch_alter_table('entity', schema=None) as batch_op:
23
+ batch_op.create_index('ix_entity_entity_type_parent', ['entity_type_id', 'parent_id'], unique=False)
24
+ batch_op.create_index('ix_entity_entity_type_project', ['entity_type_id', 'project_id'], unique=False)
25
+
26
+ # ### end Alembic commands ###
27
+
28
+
29
+ def downgrade():
30
+ # ### commands auto generated by Alembic - please adjust! ###
31
+ with op.batch_alter_table('entity', schema=None) as batch_op:
32
+ batch_op.drop_index('ix_entity_entity_type_project')
33
+ batch_op.drop_index('ix_entity_entity_type_parent')
34
+
35
+ # ### end Alembic commands ###
@@ -0,0 +1,46 @@
1
+ """add playlist_id field to notification
2
+
3
+ Revision ID: 5f1620d191af
4
+ Revises: 71d546ace0ee
5
+ Create Date: 2025-10-01 17:25:30.645533
6
+
7
+ """
8
+ from alembic import op
9
+ import sqlalchemy as sa
10
+ import sqlalchemy_utils
11
+ import sqlalchemy_utils
12
+ import uuid
13
+
14
+ # revision identifiers, used by Alembic.
15
+ revision = '5f1620d191af'
16
+ down_revision = '71d546ace0ee'
17
+ branch_labels = None
18
+ depends_on = None
19
+
20
+
21
+ def upgrade():
22
+ # ### commands auto generated by Alembic - please adjust! ###
23
+ with op.batch_alter_table('notification', schema=None) as batch_op:
24
+ batch_op.add_column(sa.Column('playlist_id', sqlalchemy_utils.types.uuid.UUIDType(binary=False), default=uuid.uuid4, nullable=True))
25
+ batch_op.alter_column('task_id',
26
+ existing_type=sa.UUID(),
27
+ nullable=True)
28
+ batch_op.drop_constraint('notification_uc', type_='unique')
29
+ batch_op.create_unique_constraint('notification_uc', ['person_id', 'author_id', 'comment_id', 'reply_id', 'playlist_id', 'type'])
30
+ batch_op.create_index(batch_op.f('ix_notification_playlist_id'), ['playlist_id'], unique=False)
31
+ batch_op.create_foreign_key(None, 'playlist', ['playlist_id'], ['id'])
32
+ # ### end Alembic commands ###
33
+
34
+
35
+ def downgrade():
36
+ # ### commands auto generated by Alembic - please adjust! ###
37
+ with op.batch_alter_table('notification', schema=None) as batch_op:
38
+ batch_op.drop_constraint(None, type_='foreignkey')
39
+ batch_op.drop_index(batch_op.f('ix_notification_playlist_id'))
40
+ batch_op.drop_constraint('notification_uc', type_='unique')
41
+ batch_op.create_unique_constraint('notification_uc', ['person_id', 'author_id', 'comment_id', 'reply_id', 'type'], postgresql_nulls_not_distinct=False)
42
+ batch_op.alter_column('task_id',
43
+ existing_type=sa.UUID(),
44
+ nullable=False)
45
+ batch_op.drop_column('playlist_id')
46
+ # ### end Alembic commands ###
@@ -0,0 +1,32 @@
1
+ """add performance indexes
2
+
3
+ Revision ID: 71d546ace0ee
4
+ Revises: 0c05b22194f3
5
+ Create Date: 2025-09-12 16:07:53.775943
6
+
7
+ """
8
+ from alembic import op
9
+ import sqlalchemy as sa
10
+ import sqlalchemy_utils
11
+
12
+
13
+ # revision identifiers, used by Alembic.
14
+ revision = '71d546ace0ee'
15
+ down_revision = '0c05b22194f3'
16
+ branch_labels = None
17
+ depends_on = None
18
+
19
+
20
+ def upgrade():
21
+ # ### commands auto generated by Alembic - please adjust! ###
22
+ with op.batch_alter_table('task', schema=None) as batch_op:
23
+ batch_op.create_index('ix_task_entity_project', ['entity_id', 'project_id'], unique=False)
24
+
25
+ # ### end Alembic commands ###
26
+
27
+
28
+ def downgrade():
29
+ # ### commands auto generated by Alembic - please adjust! ###
30
+ with op.batch_alter_table('task', schema=None) as batch_op:
31
+ batch_op.drop_index('ix_task_entity_project')
32
+ # ### end Alembic commands ###
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: zou
3
- Version: 0.20.76
3
+ Version: 0.20.77
4
4
  Summary: API to store and manage the data of your animation production
5
5
  Home-page: https://zou.cg-wire.com
6
6
  Author: CG Wire
@@ -58,10 +58,10 @@ Requires-Dist: pillow==11.3.0
58
58
  Requires-Dist: psutil==7.1.0
59
59
  Requires-Dist: psycopg[binary]==3.2.10
60
60
  Requires-Dist: pyotp==2.9.0
61
- Requires-Dist: pysaml2==7.5.2
61
+ Requires-Dist: pysaml2==7.5.3
62
62
  Requires-Dist: python-nomad==2.1.0
63
63
  Requires-Dist: python-slugify==8.0.4
64
- Requires-Dist: python-socketio==5.13.0
64
+ Requires-Dist: python-socketio==5.14.1
65
65
  Requires-Dist: pytz==2025.2
66
66
  Requires-Dist: redis==5.2.1
67
67
  Requires-Dist: requests==2.32.5
@@ -88,7 +88,7 @@ Requires-Dist: pytest==8.4.2; extra == "test"
88
88
  Provides-Extra: monitoring
89
89
  Requires-Dist: prometheus-flask-exporter==0.23.2; extra == "monitoring"
90
90
  Requires-Dist: pygelf==0.4.3; extra == "monitoring"
91
- Requires-Dist: sentry-sdk==2.39.0; extra == "monitoring"
91
+ Requires-Dist: sentry-sdk==2.40.0; extra == "monitoring"
92
92
  Provides-Extra: lint
93
93
  Requires-Dist: autoflake==2.3.1; extra == "lint"
94
94
  Requires-Dist: black==25.9.0; extra == "lint"