fix: fixed suggesting algorythm
This commit is contained in:
@@ -1,4 +1,5 @@
|
|||||||
import contextlib
|
import contextlib
|
||||||
|
from decimal import ROUND_HALF_UP, Decimal
|
||||||
from typing import Any, Self
|
from typing import Any, Self
|
||||||
from uuid import UUID
|
from uuid import UUID
|
||||||
|
|
||||||
@@ -10,7 +11,6 @@ from django.core.validators import (
|
|||||||
MinValueValidator,
|
MinValueValidator,
|
||||||
)
|
)
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from decimal import Decimal, ROUND_HALF_UP
|
|
||||||
|
|
||||||
from apps.advertiser.models import Advertiser
|
from apps.advertiser.models import Advertiser
|
||||||
from apps.campaign.validators import (
|
from apps.campaign.validators import (
|
||||||
@@ -22,6 +22,7 @@ from apps.campaign.validators import (
|
|||||||
)
|
)
|
||||||
from apps.client.models import Client
|
from apps.client.models import Client
|
||||||
from apps.core.models import BaseModel
|
from apps.core.models import BaseModel
|
||||||
|
from apps.mlscore.models import Mlscore
|
||||||
from config.errors import ConflictError, ForbiddenError
|
from config.errors import ConflictError, ForbiddenError
|
||||||
|
|
||||||
|
|
||||||
@@ -110,7 +111,7 @@ class Campaign(BaseModel):
|
|||||||
raise ValidationError(err)
|
raise ValidationError(err)
|
||||||
except Campaign.DoesNotExist:
|
except Campaign.DoesNotExist:
|
||||||
if self.start_date < current_date:
|
if self.start_date < current_date:
|
||||||
raise ValidationError(err)
|
raise ValidationError(err) from None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def ad_id(self) -> UUID:
|
def ad_id(self) -> UUID:
|
||||||
@@ -277,101 +278,114 @@ class Campaign(BaseModel):
|
|||||||
| models.Q(age_from__isnull=True)
|
| models.Q(age_from__isnull=True)
|
||||||
) & (models.Q(age_to__gte=client.age) | models.Q(age_to__isnull=True))
|
) & (models.Q(age_to__gte=client.age) | models.Q(age_to__isnull=True))
|
||||||
|
|
||||||
queryset = (
|
return (
|
||||||
cls.objects.filter(
|
cls.objects.filter(
|
||||||
date_filter,
|
date_filter,
|
||||||
location_filter,
|
location_filter,
|
||||||
gender_filter,
|
gender_filter,
|
||||||
age_filter,
|
age_filter,
|
||||||
)
|
)
|
||||||
.prefetch_related("clicks", "impressions")
|
|
||||||
.select_related("advertiser")
|
.select_related("advertiser")
|
||||||
|
.prefetch_related("clicks", "impressions", "advertiser__mlscores")
|
||||||
)
|
)
|
||||||
|
|
||||||
return queryset
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def suggest(cls, client: Client) -> None | Self:
|
def suggest(cls, client: Client) -> Self:
|
||||||
available_campaigns = Campaign.get_available_campaigns(client)
|
base_campaigns = cls.get_available_campaigns(client)
|
||||||
if not available_campaigns or available_campaigns == []:
|
if not base_campaigns or base_campaigns == []:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
campaign_ids = [c.id for c in available_campaigns]
|
advertiser_ids = list({c.advertiser_id for c in base_campaigns})
|
||||||
|
ml_scores = Mlscore.objects.filter(
|
||||||
|
client=client, advertiser_id__in=advertiser_ids
|
||||||
|
).values("advertiser_id", "score")
|
||||||
|
ml_dict = {m["advertiser_id"]: m["score"] for m in ml_scores}
|
||||||
|
|
||||||
impressions_counts = (
|
campaigns = list(
|
||||||
CampaignImpression.objects.filter(campaign_id__in=campaign_ids)
|
base_campaigns.annotate(
|
||||||
.values("campaign_id")
|
impressions_count=models.Count("impressions"),
|
||||||
.annotate(total=models.Count("id"))
|
clicks_count=models.Count("clicks"),
|
||||||
)
|
)
|
||||||
impressions_dict = {
|
|
||||||
item["campaign_id"]: item["total"] for item in impressions_counts
|
|
||||||
}
|
|
||||||
|
|
||||||
clicks_counts = (
|
|
||||||
CampaignClick.objects.filter(campaign_id__in=campaign_ids)
|
|
||||||
.values("campaign_id")
|
|
||||||
.annotate(total=models.Count("id"))
|
|
||||||
)
|
)
|
||||||
clicks_dict = {
|
campaign_ids = [c.id for c in campaigns]
|
||||||
item["campaign_id"]: item["total"] for item in clicks_counts
|
|
||||||
}
|
|
||||||
|
|
||||||
existing_impressions = CampaignImpression.objects.filter(
|
client_impressions = set(
|
||||||
|
CampaignImpression.objects.filter(
|
||||||
client=client, campaign_id__in=campaign_ids
|
client=client, campaign_id__in=campaign_ids
|
||||||
).values_list("campaign_id", flat=True)
|
).values_list("campaign_id", flat=True)
|
||||||
existing_impressions_set = set(existing_impressions)
|
)
|
||||||
|
client_clicks = set(
|
||||||
advertisers = {c.advertiser_id for c in available_campaigns}
|
CampaignClick.objects.filter(
|
||||||
ml_scores = models.Mlscore.objects.filter(
|
client=client, campaign_id__in=campaign_ids
|
||||||
client=client, advertiser_id__in=advertisers
|
).values_list("campaign_id", flat=True)
|
||||||
).values("advertiser_id", "score")
|
)
|
||||||
ml_score_dict = {ms["advertiser_id"]: ms["score"] for ms in ml_scores}
|
|
||||||
max_ml = max(ml_score_dict.values(), default=0)
|
|
||||||
|
|
||||||
valid_campaigns = []
|
|
||||||
for campaign in available_campaigns:
|
|
||||||
current_impressions = impressions_dict.get(campaign.id, 0)
|
|
||||||
if current_impressions >= campaign.impressions_limit:
|
|
||||||
continue
|
|
||||||
if campaign.id in existing_impressions_set:
|
|
||||||
continue
|
|
||||||
valid_campaigns.append(campaign)
|
|
||||||
|
|
||||||
prioritized = []
|
prioritized = []
|
||||||
for campaign in valid_campaigns:
|
ml_values = []
|
||||||
ml_score = ml_score_dict.get(campaign.advertiser_id, 0)
|
profit_values = []
|
||||||
|
|
||||||
remaining_impressions = (
|
for campaign in campaigns:
|
||||||
campaign.impressions_limit
|
if campaign.impressions_count >= campaign.impressions_limit:
|
||||||
- impressions_dict.get(campaign.id, 0)
|
continue
|
||||||
)
|
|
||||||
weight_imp = (
|
|
||||||
remaining_impressions / campaign.impressions_limit
|
|
||||||
if campaign.impressions_limit > 0
|
|
||||||
else 1.0
|
|
||||||
)
|
|
||||||
|
|
||||||
current_clicks = clicks_dict.get(campaign.id, 0)
|
ml_score = ml_dict.get(campaign.advertiser_id, 0)
|
||||||
remaining_clicks = campaign.clicks_limit - current_clicks
|
ml_values.append(ml_score)
|
||||||
if remaining_clicks <= 0:
|
|
||||||
cpc_contribution = 0.0
|
has_impression = campaign.id in client_impressions
|
||||||
|
has_click = campaign.id in client_clicks
|
||||||
|
|
||||||
|
if has_impression:
|
||||||
|
profit = campaign.cost_per_click if not has_click else 0
|
||||||
else:
|
else:
|
||||||
click_availability = (
|
profit = campaign.cost_per_impression + campaign.cost_per_click
|
||||||
(remaining_clicks / campaign.clicks_limit)
|
print(profit)
|
||||||
if campaign.clicks_limit > 0
|
if profit <= 0:
|
||||||
else 1.0
|
continue
|
||||||
|
|
||||||
|
profit_values.append(profit)
|
||||||
|
|
||||||
|
remaining_imp = (
|
||||||
|
campaign.impressions_limit - campaign.impressions_count
|
||||||
)
|
)
|
||||||
prob_click = (ml_score / max_ml) if max_ml != 0 else 0.0
|
capacity_ratio = (
|
||||||
cpc_contribution = (
|
remaining_imp / campaign.impressions_limit
|
||||||
campaign.cost_per_click * prob_click * click_availability
|
if campaign.impressions_limit > 0
|
||||||
|
else 1
|
||||||
)
|
)
|
||||||
|
|
||||||
expected_profit = campaign.cost_per_impression + cpc_contribution
|
prioritized.append(
|
||||||
priority = ml_score * expected_profit * weight_imp
|
(
|
||||||
prioritized.append((campaign, priority))
|
campaign,
|
||||||
|
{
|
||||||
|
"profit": profit,
|
||||||
|
"ml": ml_score,
|
||||||
|
"capacity": 1 - capacity_ratio,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
prioritized.sort(key=lambda x: -x[1])
|
max_ml = max(ml_values) if ml_values else 1
|
||||||
return prioritized[0]
|
max_profit = max(profit_values) if profit_values else 1
|
||||||
|
min_profit = min(profit_values) if profit_values else 0
|
||||||
|
profit_range = (
|
||||||
|
max_profit - min_profit if max_profit != min_profit else 1
|
||||||
|
)
|
||||||
|
|
||||||
|
print(prioritized, max_ml, max_profit, min_profit, profit_range)
|
||||||
|
|
||||||
|
final_list = []
|
||||||
|
for campaign, metrics in prioritized:
|
||||||
|
norm_profit = (metrics["profit"] - min_profit) / profit_range
|
||||||
|
norm_ml = metrics["ml"] / max_ml if max_ml > 0 else 0
|
||||||
|
|
||||||
|
priority = (
|
||||||
|
0.5 * norm_profit + 0.25 * norm_ml + 0.15 * metrics["capacity"]
|
||||||
|
)
|
||||||
|
|
||||||
|
final_list.append((campaign, priority))
|
||||||
|
|
||||||
|
final_list.sort(key=lambda x: -x[1])
|
||||||
|
return final_list[0][0] if len(final_list) >= 1 else None
|
||||||
|
|
||||||
|
|
||||||
class CampaignImpression(BaseModel):
|
class CampaignImpression(BaseModel):
|
||||||
|
|||||||
Reference in New Issue
Block a user