wbcrm 2.2.4__py2.py3-none-any.whl → 2.2.6__py2.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.

Potentially problematic release.


This version of wbcrm might be problematic. Click here for more details.

@@ -0,0 +1,33 @@
1
+ from typing import TYPE_CHECKING
2
+
3
+ from langchain_core.messages import HumanMessage, SystemMessage
4
+ from pydantic import BaseModel, Field
5
+ from wbcore.contrib.ai.llm.config import LLMConfig
6
+
7
+ if TYPE_CHECKING:
8
+ from wbcrm.models import Activity
9
+
10
+
11
+ def analyze_activity_prompt(activity: "Activity"):
12
+ return [
13
+ SystemMessage(
14
+ content="You are a manager in a company and there are activities, both internal as well as external, meaning between service providers, clients and ourselves. We want to analyze these activities in regards of their sentiment and summarize them in English. The summary should be a very short bottom-line focussed text."
15
+ ),
16
+ HumanMessage(
17
+ content=f"title={activity.title}, description={activity.description}, period={activity.period}, participants={activity.participants.all()}, companies={activity.companies.all()}, review={activity.result}"
18
+ ),
19
+ ]
20
+
21
+
22
+ class ActivityLLMResponseModel(BaseModel):
23
+ heat: int = Field(..., ge=1, le=4, description="The sentiment heat.")
24
+ summary: str = Field(..., description="A summary of the activity in English.")
25
+
26
+
27
+ analyze_activity = LLMConfig["Activity"](
28
+ key="analyze",
29
+ prompt=analyze_activity_prompt,
30
+ on_save=True,
31
+ on_condition=lambda instance: instance.status == "REVIEWED",
32
+ output_model=ActivityLLMResponseModel,
33
+ )
@@ -0,0 +1,54 @@
1
+ from typing import TYPE_CHECKING
2
+
3
+ from langchain_core.messages import HumanMessage, SystemMessage
4
+ from pydantic import BaseModel, Field
5
+ from wbcore.contrib.ai.llm.config import LLMConfig
6
+ from wbcrm.models.activities import Activity
7
+
8
+ if TYPE_CHECKING:
9
+ from wbcrm.models import Account
10
+
11
+
12
+ def analyze_relationship_prompt(account: "Account"):
13
+ messages = [
14
+ SystemMessage(
15
+ content="Based on the recent interactions/activities, current relationship health, and planned actions with the customer, provide a status score (1 to 5, where 1 is cold and 5 is hot), a summary of the relationship, and recommended next steps for maintaining or improving the relationship. Keep in mind that more information might become available later to refine these insights. Please include all information that is relevant to the relationship in the summary. Regarding the activities/interactions, older interactions are less relevant than recent ones."
16
+ ),
17
+ HumanMessage(
18
+ content=f"The title of the account: {account.title}",
19
+ ),
20
+ ]
21
+ if account.owner:
22
+ for activity in Activity.objects.filter(companies__id__in=[account.owner.id], status="REVIEWED"):
23
+ messages.append(
24
+ HumanMessage(
25
+ content=f"Activity: {activity.summary}, with sentiment: {activity.heat}, period: {activity.period}",
26
+ )
27
+ )
28
+
29
+ return messages
30
+
31
+
32
+ class AccountRelationshipResponseModel(BaseModel):
33
+ relationship_status: int = Field(
34
+ ...,
35
+ ge=1,
36
+ le=5,
37
+ description="Rate the customer relationship status from 1 to 5. 1 being the cold and 5 being the hot.",
38
+ )
39
+ relationship_summary: str = Field(
40
+ ...,
41
+ description="Briefly summarize the current state of the relationship and recent interactions. Also include any additional information that might be relevant.",
42
+ )
43
+ action_plan: str = Field(
44
+ ..., description="Provide the next recommended actions or steps to engage with the customer."
45
+ )
46
+
47
+
48
+ analyze_relationship = LLMConfig["Account"](
49
+ key="analyze_relationship",
50
+ prompt=analyze_relationship_prompt,
51
+ on_save=True,
52
+ on_condition=lambda act: (act.status == act.Status.OPEN) and act.owner is not None and act.is_root_node(),
53
+ output_model=AccountRelationshipResponseModel,
54
+ )
@@ -0,0 +1,110 @@
1
+ from datetime import datetime
2
+ from io import BytesIO
3
+
4
+ import xlsxwriter
5
+ from celery import shared_task
6
+ from django.conf import settings
7
+ from django.core.mail import EmailMultiAlternatives
8
+ from django.db.models import Q
9
+ from django.template.loader import get_template
10
+ from django.utils import timezone
11
+ from django.utils.html import strip_tags
12
+ from django.utils.translation import gettext as _
13
+ from wbcore.contrib.directory.models import Person
14
+ from wbcrm.models import Activity, ActivityType
15
+ from xlsxwriter.utility import xl_rowcol_to_cell
16
+
17
+
18
+ def set_cell(worksheet, row, col, value, width_cols, base_format):
19
+ cell = xl_rowcol_to_cell(row, col)
20
+ if value:
21
+ worksheet.write_string(cell, value, base_format)
22
+ if len(value) > width_cols[col]:
23
+ width_cols[col] = len(value)
24
+
25
+
26
+ @shared_task
27
+ def create_report_and_send(profile_id, employee_id, start_date=None, end_date=None):
28
+ employee = Person.objects.get(id=employee_id)
29
+ profile = Person.objects.get(id=profile_id)
30
+ if end_date is None:
31
+ end_date = timezone.now()
32
+ if start_date is None:
33
+ start_date = datetime(year=end_date.year, month=1, day=1, hour=0, minute=0, second=0)
34
+
35
+ report = create_report(employee_id, start_date, end_date)
36
+
37
+ html = get_template("email/activity_report.html")
38
+
39
+ context = {"profile": profile, "employee": employee}
40
+ html_content = html.render(context)
41
+
42
+ msg = EmailMultiAlternatives(
43
+ _("Report"), strip_tags(html_content), settings.DEFAULT_FROM_EMAIL, [profile.user_account.email]
44
+ )
45
+ msg.attach_alternative(html_content, "text/html")
46
+ title = _("all")
47
+ if start_date and end_date:
48
+ title = "{}_{}".format(start_date.strftime("%m/%d/%Y-%H:%M:%S"), end_date.strftime("%m/%d/%Y-%H:%M:%S"))
49
+ msg.attach(
50
+ "report_{}_{}.xlsx".format(employee.computed_str, title),
51
+ report.read(),
52
+ "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
53
+ )
54
+ msg.send()
55
+
56
+
57
+ def create_report(employee, start_date, end_date):
58
+ output = BytesIO()
59
+ workbook = xlsxwriter.Workbook(output, {"in_memory": True})
60
+
61
+ base_format = workbook.add_format({"font_name": "Liberation Sans", "font_size": 10})
62
+ bold_format = workbook.add_format({"font_name": "Liberation Sans", "font_size": 10, "bold": True})
63
+ related_activities = Activity.objects.filter(
64
+ Q(participants__id=employee) | Q(creator__id=employee) | Q(assigned_to__id=employee)
65
+ )
66
+ activities = Activity.get_inrange_activities(related_activities, start_date, end_date)
67
+ # HERE STARTS THE FIRST WORKSHEET
68
+ for type in ActivityType.objects.all():
69
+ width_cols = [5, 5, 16, 8, 7, 11, 12]
70
+ activities_type = activities.filter(type=type.id)
71
+ worksheet = workbook.add_worksheet(type.title)
72
+ worksheet.write_string(0, 0, _("Title"), bold_format)
73
+ worksheet.write_string(0, 1, _("Start"), bold_format)
74
+ worksheet.write_string(0, 2, _("Duration (hours)"), bold_format)
75
+ worksheet.write_string(0, 3, _("Location"), bold_format)
76
+ worksheet.write_string(0, 4, _("Creator"), bold_format)
77
+ worksheet.write_string(0, 5, _("Assigned to"), bold_format)
78
+ worksheet.write_string(0, 6, _("Participants"), bold_format)
79
+
80
+ worksheet.write_string(2, 0, _("Total"), bold_format)
81
+ worksheet.write_string(2, 1, str(len(activities_type.all())), bold_format)
82
+
83
+ for row, activity in enumerate(activities_type.all(), start=4):
84
+ hours = (activity.period.upper - activity.period.lower).total_seconds() / 3600
85
+ duration = format(hours, ".2f")
86
+ creator = activity.creator
87
+ creator_name = ""
88
+ if creator:
89
+ creator_name = creator.computed_str
90
+ assigned_to = activity.assigned_to
91
+ assigned_to_name = ""
92
+ if assigned_to:
93
+ assigned_to_name = assigned_to.computed_str
94
+ participants = ", ".join([participant.computed_str for participant in activity.participants.all()])
95
+
96
+ set_cell(worksheet, row, 0, activity.title, width_cols, base_format)
97
+ set_cell(worksheet, row, 1, f"{activity.period.lower:%d.%m.%Y}", width_cols, base_format)
98
+ set_cell(worksheet, row, 2, duration, width_cols, base_format)
99
+ set_cell(worksheet, row, 2, activity.location, width_cols, base_format)
100
+ set_cell(worksheet, row, 4, creator_name, width_cols, base_format)
101
+ set_cell(worksheet, row, 5, assigned_to_name, width_cols, base_format)
102
+ set_cell(worksheet, row, 6, participants, width_cols, base_format)
103
+
104
+ for col, max_width in enumerate(width_cols):
105
+ if max_width > 300:
106
+ max_width = 300
107
+ worksheet.set_column(col, col, max_width)
108
+ workbook.close()
109
+ output.seek(0)
110
+ return output
@@ -0,0 +1,86 @@
1
+ # Activities
2
+ A list of every activity saved in the database. Will be filtered by activities that happen in the timeframe from a month ago to one week in the future by default. Will be ordered by the date an activity was last edited.
3
+
4
+ ## Workflow:
5
+ An activity can have one of five different states that can be switched between by clicking on the corresponding buttons. These are displayed in the context menu if you right click the list row or at the top of the instance.
6
+
7
+ ![Activity Workflow]()
8
+
9
+
10
+ ### Planned:
11
+ The initial status after creating a new activity. From here you can either cancel, finish or directly review the activity which creates a popup where you can write your review.
12
+
13
+ ### Finished:
14
+ The activity's status after it is over. From here you can review the activity.
15
+
16
+ ### Canceled:
17
+ After canceling the activity this is its status.
18
+
19
+ ### Reviewed:
20
+ After review this is the final status of the activity.
21
+
22
+ ## Columns:
23
+ Each column title has three lines on the right if you hover over it. Click on them to show options for that column. The second tab of the options menu will allow you to filter the column, the third to completely hide entire columns. Click on anywhere else on the column to order it, cycling between ascending, descending and no ordering. Hold shift while clicking to order multiple columns with individual weights.
24
+
25
+ ### Type:
26
+ The type of the activity. Activity types can be added, edited and deleted by managers under CRM > Administration > Activity Types.
27
+
28
+ ### Title:
29
+ The activity's name.
30
+
31
+ ### Period:
32
+ The timeframe where the activity takes place.
33
+
34
+ ### Participants:
35
+ A list of persons participating in the activity. Hover over a name to display more information.
36
+
37
+ ### Companies:
38
+ A list of companies participating in the activity. Hover over a name to display more information.
39
+
40
+ ### Groups:
41
+ A list of groups participating in the activity. Groups can consist of persons and companies which will all be automatically added to the activity's participants and companies fields. Groups can be added, edited and deleted under CRM > Administration > Groups.
42
+
43
+ ### Edited:
44
+ The date at which the activity was last edited.
45
+
46
+ ### Created:
47
+ The date at which the activity was originally created.
48
+
49
+ ### Description:
50
+ A description of what the activity is about. Hover over it to display the full description.
51
+
52
+ ### Review:
53
+ A written review of the activity which can be created by anyone who can see the activity.
54
+
55
+ ### Latest Reviewer:
56
+ The last person who updated the activity's review. Hover over the name to display more details about him/her.
57
+
58
+ ## Legend:
59
+ Click on a status in the legend to filter the list. Repeating activities have a special symbol to the left of their column.
60
+
61
+ ## Filters:
62
+ Filters are accessed by clicking on the symbol in the top left corner of the window.
63
+
64
+ ### Status:
65
+ Filter activities by a specific status they're in, smiliarly to what clicking on the legend does.
66
+
67
+ ### Frequency:
68
+ Filter activities by if and how often an activity is repeated.
69
+
70
+ ### Conference Room:
71
+ Filter activities by if they take place in the conference room.
72
+
73
+ ### Clients of:
74
+ Display only activities involving clients of the selected person.
75
+
76
+ ### Visibility:
77
+ Filter activities by their visibility: Public, private or confidential.
78
+
79
+ ### Importance:
80
+ Filter activities by their importance. The default importance value for activities is "Low".
81
+
82
+ ### Only Recent:
83
+ Display activities that happen in the timeframe from a month ago to one week in the future. Will be active by default.
84
+
85
+ ## Search Field:
86
+ Typing in the search field allows to filter the activities by name, description and review.
@@ -0,0 +1,20 @@
1
+ # Activity Types
2
+ This is a list of every activity type in the database where you can modify, delete and add new activity types. Typing in the search field filters the activity types by name.
3
+
4
+ ## Columns:
5
+ Each column title has three lines on the right if you hover over it. Click on them to show options for that column. The second tab of the options menu will allow you to filter the column, the third to completely hide entire columns. Click on anywhere else on the column to order it, cycling between ascending, descending and no ordering. Hold shift while clicking to order multiple columns with individual weights.
6
+
7
+ ### Name:
8
+ The name of the activity type.
9
+
10
+ ### Icon:
11
+ The icon representing the activity that is displayed in the calendar.
12
+
13
+ ### Color:
14
+ The hexadecimal color code used to plot activity types. Must be unique.
15
+
16
+ ### Multiplier:
17
+ Every activity type has a multiplier for the activity heat calculation. Multipliers range from low (2, i.e. e-mail) and medium (3, i.e. call) to high (4, i.e. meeting).
18
+
19
+ ### Is Default:
20
+ If true sets the default value for type in new activities to be this instance.
@@ -0,0 +1,2 @@
1
+ # Groups
2
+ This is a list of every group in the database where you can modify, delete and add new groups. Typing in the search field or the column filter accessible from the three lines at the end of the column title filters the groups by name. Click on the column title to order it.
@@ -0,0 +1,11 @@
1
+ # Products
2
+ This is a list of every CRM product in the database where you can modify, delete and add new products. Typing in the search field filters the products by name.
3
+
4
+ ## Columns:
5
+ Each column title has three lines on the right if you hover over it. Click on them to show options for that column. The second tab of the options menu will allow you to filter the column, the third to completely hide entire columns. Click on anywhere else on the column to order it, cycling between ascending, descending and no ordering. Hold shift while clicking to order multiple columns with individual weights.
6
+
7
+ ### Title:
8
+ The name of the product.
9
+
10
+ ### Is Competitor:
11
+ Indicates if this is a competitor's product.
@@ -0,0 +1,75 @@
1
+ from datetime import timedelta
2
+
3
+ from celery import shared_task
4
+ from dynamic_preferences.registries import global_preferences_registry
5
+ from googleapiclient.discovery import Resource
6
+ from psycopg.types.range import TimestamptzRange
7
+ from wbcore.contrib.authentication.models import User
8
+ from wbcrm.models import Activity
9
+
10
+ from ...typing_informations import GoogleEventType
11
+ from ...utils import GoogleSyncUtils
12
+ from .update import update_activity_participant
13
+
14
+
15
+ @shared_task(queue="synchronization")
16
+ def create_internal_activity_based_on_google_event(
17
+ event: GoogleEventType, user: User, service: Resource, parent_occurrence: Activity | None = None, is_instance=False
18
+ ):
19
+ """
20
+ A method for creating a "Workbench" activity based on a "Google" event. If the google event is a recurring event, this method will also create the corresponding workbench activities.
21
+
22
+ :param event: A google event body.
23
+ :param user: The current workbench user.
24
+ :param service: Thee google Resource.
25
+ :param parent_occurrence: The parent activity. This is only used when the activity is part of a recurring chain. Per default it is None.
26
+ :param is_instance: True, when the event is a google instance (A instance is an event that is part of a recurring chain)
27
+ """
28
+
29
+ external_id = event["id"]
30
+ if event.get("status") == "cancelled" or Activity.objects.filter(external_id=event["id"]).exists():
31
+ return
32
+
33
+ event_creator = event.get("organizer", {})
34
+ event_creator_mail = event_creator.get("email", "")
35
+ event_creator_displayed_name = event_creator.get("displayName", "")
36
+ creator = GoogleSyncUtils.get_or_create_person(event_creator_mail, event_creator_displayed_name)
37
+ event_start, event_end = GoogleSyncUtils.get_start_and_end(event)
38
+ if event_start == event_end:
39
+ event_end = event_end + timedelta(seconds=1)
40
+ period = TimestamptzRange(event_start, event_end) # type: ignore
41
+ all_day: bool = True if event["start"].get("date") else False
42
+ metadata = {"google_backend": {"instance": event}} if is_instance else {"google_backend": {"event": event}}
43
+
44
+ act = Activity.objects.create(
45
+ external_id=external_id,
46
+ title=event.get("summary", "(No Subject)"),
47
+ assigned_to=creator,
48
+ creator=creator,
49
+ description=event.get("description", ""),
50
+ start=event_start,
51
+ end=event_end,
52
+ period=period,
53
+ all_day=all_day,
54
+ location=event.get("location"),
55
+ visibility=GoogleSyncUtils.convert_event_visibility_to_activity_visibility(event.get("visibility", "")),
56
+ metadata=metadata,
57
+ parent_occurrence=parent_occurrence,
58
+ )
59
+
60
+ update_activity_participant(event, act)
61
+
62
+ if event.get("recurrence") and not is_instance:
63
+ instances: dict = service.events().instances(calendarId=user.email, eventId=event["id"]).execute()
64
+ instance_items: list[GoogleEventType] = instances["items"]
65
+ global_preferences = global_preferences_registry.manager()
66
+ max_list_length: int = global_preferences["wbcrm__recurrence_maximum_count"]
67
+ instance_items = instance_items[:max_list_length]
68
+
69
+ for item in instance_items:
70
+ if item["start"] == act.metadata["google_backend"].get("event", {}).get("start"):
71
+ updated_metadata: dict = act.metadata
72
+ updated_metadata["google_backend"] |= {"instance": item}
73
+ Activity.objects.filter(id=act.id).update(metadata=updated_metadata)
74
+ else:
75
+ create_internal_activity_based_on_google_event(item, user, service, act, True)
@@ -0,0 +1,78 @@
1
+ from django.db.models import QuerySet
2
+ from dynamic_preferences.registries import global_preferences_registry
3
+ from googleapiclient.discovery import Resource
4
+ from wbcrm.models import Activity
5
+
6
+
7
+ def cancel_or_delete_activity(activity: "Activity") -> None:
8
+ # Activity is cancelled rather than disable if global preference is True
9
+ if global_preferences_registry.manager()["wbactivity_sync__sync_cancelled_activity"]:
10
+ Activity.objects.filter(id=activity.id).update(status=Activity.Status.CANCELLED)
11
+ else:
12
+ activity.delete()
13
+
14
+
15
+ def cancel_or_delete_activity_queryset(activity_qs: QuerySet["Activity"]) -> None:
16
+ # Activities are cancelled rather than delete if global preference is True
17
+ if global_preferences_registry.manager()["wbactivity_sync__sync_cancelled_activity"]:
18
+ activity_qs.update(status=Activity.Status.CANCELLED)
19
+ else:
20
+ activity_qs.delete()
21
+
22
+
23
+ def delete_single_activity(activity: Activity):
24
+ """
25
+ Deletes the activity that has the corresponding external ID
26
+ """
27
+ Activity.objects.filter(id=activity.id).update(external_id=None)
28
+ cancel_or_delete_activity(activity)
29
+
30
+
31
+ def delete_recurring_activity(activity: Activity, event: dict, user_mail: str, service: Resource):
32
+ """
33
+ Handles the deletion of recurring activities (either a single activity, a certain number of activities or all activities),
34
+ based on the changes done to the google event.
35
+
36
+ :param event: The google event dict that was deleted.
37
+ :param activity: The corresponding workbench activity.
38
+ :param user_mail: The e-mail address of the current user.
39
+ :param service: The google service to interact with googles resources.
40
+ """
41
+
42
+ parent_occurrence = activity.parent_occurrence if activity.parent_occurrence else activity
43
+ metadata_event = parent_occurrence.metadata["google_backend"].get(
44
+ "event", parent_occurrence.metadata["google_backend"].get("instance", {})
45
+ )
46
+ if event.get("recurringEventId"):
47
+ # Delete single event in event chain
48
+ if activity.status not in [Activity.Status.REVIEWED, Activity.Status.FINISHED]:
49
+ cancel_or_delete_activity(activity)
50
+
51
+ elif event.get("recurrence") != metadata_event.get("recurrence"):
52
+ # Delete all events after a certain event in the event chain
53
+
54
+ event_instances: dict = service.events().instances(calendarId=user_mail, eventId=event["id"]).execute()
55
+ external_id_list: list = [instance["id"] for instance in event_instances["items"]]
56
+ activities_to_remove = Activity.objects.filter(
57
+ metadata__google_backend__instance__recurringEventId__startswith=event["id"],
58
+ status=Activity.Status.PLANNED,
59
+ ).exclude(metadata__google_backend__instance__id__in=external_id_list)
60
+ first_in_list = activities_to_remove.order_by("start").first()
61
+ metadata = parent_occurrence.metadata
62
+ metadata["google_backend"] |= {"event": event}
63
+ Activity.objects.filter(id=parent_occurrence.id).update(metadata=metadata)
64
+ if (
65
+ first_in_list
66
+ and first_in_list.id == parent_occurrence.id
67
+ and parent_occurrence.status not in [Activity.Status.REVIEWED, Activity.Status.FINISHED]
68
+ ):
69
+ cancel_or_delete_activity(parent_occurrence)
70
+ cancel_or_delete_activity_queryset(activities_to_remove)
71
+
72
+ else:
73
+ # Delete all events in event chain
74
+ activities_to_cancel = Activity.objects.filter(
75
+ parent_occurrence=parent_occurrence, status=Activity.Status.PLANNED
76
+ )
77
+ cancel_or_delete_activity_queryset(activities_to_cancel)
78
+ cancel_or_delete_activity(activity)