Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Refactor promoted groups to use APIChoicesWithNone #23051

Merged
merged 3 commits into from
Feb 11, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 4 additions & 4 deletions src/olympia/abuse/tests/test_actions.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@
version_factory,
)
from olympia.constants.abuse import DECISION_ACTIONS
from olympia.constants.promoted import RECOMMENDED
from olympia.constants.promoted import PROMOTED_GROUP_CHOICES
from olympia.core import set_user
from olympia.ratings.models import Rating
from olympia.reviewers.models import ReviewActionReason
Expand Down Expand Up @@ -464,7 +464,7 @@ def test_should_hold_action(self):
assert action.should_hold_action() is False
addon = addon_factory(users=[self.user])
assert action.should_hold_action() is False
self.make_addon_promoted(addon, RECOMMENDED)
self.make_addon_promoted(addon, PROMOTED_GROUP_CHOICES.RECOMMENDED)
assert action.should_hold_action() is True

self.user.banned = datetime.now()
Expand Down Expand Up @@ -927,7 +927,7 @@ def test_should_hold_action(self):
action = self.ActionClass(self.decision)
assert action.should_hold_action() is False

self.make_addon_promoted(self.addon, RECOMMENDED)
self.make_addon_promoted(self.addon, PROMOTED_GROUP_CHOICES.RECOMMENDED)
assert action.should_hold_action() is True

self.addon.status = amo.STATUS_DISABLED
Expand Down Expand Up @@ -1342,7 +1342,7 @@ def test_should_hold_action(self):

AddonUser.objects.create(addon=self.rating.addon, user=self.rating.user)
assert action.should_hold_action() is False
self.make_addon_promoted(self.rating.addon, RECOMMENDED)
self.make_addon_promoted(self.rating.addon, PROMOTED_GROUP_CHOICES.RECOMMENDED)
assert action.should_hold_action() is False
self.rating.update(
reply_to=Rating.objects.create(
Expand Down
8 changes: 4 additions & 4 deletions src/olympia/abuse/tests/test_cinder.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@
ILLEGAL_CATEGORIES,
ILLEGAL_SUBCATEGORIES,
)
from olympia.constants.promoted import NOT_PROMOTED, NOTABLE, RECOMMENDED
from olympia.constants.promoted import PROMOTED_GROUP_CHOICES
from olympia.ratings.models import Rating
from olympia.reviewers.models import NeedsHumanReview
from olympia.users.models import UserProfile
Expand Down Expand Up @@ -390,7 +390,7 @@ def test_build_report_payload_promoted_recommended(self):
privacy_policy='Söme privacy policy',
version_kw={'release_notes': 'Søme release notes'},
)
self.make_addon_promoted(addon, group=RECOMMENDED)
self.make_addon_promoted(addon, group_id=PROMOTED_GROUP_CHOICES.RECOMMENDED)
message = ' bad addon!'
cinder_addon = self.CinderClass(addon)
encoded_message = cinder_addon.get_str(message)
Expand Down Expand Up @@ -457,7 +457,7 @@ def test_build_report_payload_promoted_notable(self):
privacy_policy='Söme privacy policy',
version_kw={'release_notes': 'Søme release notes'},
)
self.make_addon_promoted(addon, group=NOTABLE)
self.make_addon_promoted(addon, group_id=PROMOTED_GROUP_CHOICES.NOTABLE)
message = ' bad addon!'
cinder_addon = self.CinderClass(addon)
encoded_message = cinder_addon.get_str(message)
Expand Down Expand Up @@ -515,7 +515,7 @@ def test_build_report_payload_promoted_notable(self):
},
}

self.make_addon_promoted(addon, NOT_PROMOTED)
self.make_addon_promoted(addon, PROMOTED_GROUP_CHOICES.NOT_PROMOTED)
data = cinder_addon.build_report_payload(
report=CinderReport(abuse_report), reporter=None
)
Expand Down
18 changes: 13 additions & 5 deletions src/olympia/abuse/tests/test_models.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@
ILLEGAL_CATEGORIES,
ILLEGAL_SUBCATEGORIES,
)
from olympia.constants.promoted import RECOMMENDED
from olympia.constants.promoted import PROMOTED_GROUP_CHOICES
from olympia.core import set_user
from olympia.ratings.models import Rating
from olympia.reviewers.models import NeedsHumanReview
Expand Down Expand Up @@ -2757,7 +2757,9 @@ def _test_execute_action_disable_addon_outcome(self, decision):

def test_execute_action_disable_addon_held(self):
addon = addon_factory(users=[user_factory()])
self.make_addon_promoted(addon, RECOMMENDED, approve_version=True)
self.make_addon_promoted(
addon, PROMOTED_GROUP_CHOICES.RECOMMENDED, approve_version=True
)
decision = ContentDecision.objects.create(
addon=addon,
action=DECISION_ACTIONS.AMO_DISABLE_ADDON,
Expand Down Expand Up @@ -2802,7 +2804,9 @@ def _test_execute_action_reject_version_outcome(self, decision):
def test_execute_action_reject_version_held(self):
addon = addon_factory(users=[user_factory()])
version = addon.current_version
self.make_addon_promoted(addon, RECOMMENDED, approve_version=True)
self.make_addon_promoted(
addon, PROMOTED_GROUP_CHOICES.RECOMMENDED, approve_version=True
)
decision = ContentDecision.objects.create(
addon=addon,
action=DECISION_ACTIONS.AMO_REJECT_VERSION_ADDON,
Expand Down Expand Up @@ -2899,7 +2903,9 @@ def test_execute_action_reject_version_delayed(self):
def test_execute_action_reject_version_delayed_held(self):
addon = addon_factory(users=[user_factory()])
version = addon.current_version
self.make_addon_promoted(addon, RECOMMENDED, approve_version=True)
self.make_addon_promoted(
addon, PROMOTED_GROUP_CHOICES.RECOMMENDED, approve_version=True
)
decision = ContentDecision.objects.create(
addon=addon,
action=DECISION_ACTIONS.AMO_REJECT_VERSION_WARNING_ADDON,
Expand Down Expand Up @@ -3134,7 +3140,9 @@ def test_execute_action_delete_rating_held(self):
action=DECISION_ACTIONS.AMO_DELETE_RATING,
reviewer_user=self.reviewer_user,
)
self.make_addon_promoted(rating.addon, RECOMMENDED, approve_version=True)
self.make_addon_promoted(
rating.addon, PROMOTED_GROUP_CHOICES.RECOMMENDED, approve_version=True
)
assert decision.action_date is None
mail.outbox.clear()

Expand Down
4 changes: 2 additions & 2 deletions src/olympia/addons/indexers.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
from olympia import amo
from olympia.amo.celery import create_chunked_tasks_signatures
from olympia.amo.utils import attach_trans_dict, to_language
from olympia.constants.promoted import RECOMMENDED
from olympia.constants.promoted import PROMOTED_GROUP_CHOICES
from olympia.constants.search import SEARCH_LANGUAGE_TO_ANALYZER
from olympia.search.utils import create_index
from olympia.versions.compare import version_int
Expand Down Expand Up @@ -663,7 +663,7 @@ def extract_document(cls, obj):
data['has_privacy_policy'] = bool(obj.privacy_policy)

data['is_recommended'] = bool(
obj.promoted and obj.promoted.group == RECOMMENDED
obj.promoted and obj.promoted.group_id == PROMOTED_GROUP_CHOICES.RECOMMENDED
)

data['previews'] = [
Expand Down
17 changes: 13 additions & 4 deletions src/olympia/addons/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -57,7 +57,10 @@
)
from olympia.constants.browsers import BROWSERS
from olympia.constants.categories import CATEGORIES_BY_ID
from olympia.constants.promoted import NOT_PROMOTED, RECOMMENDED
from olympia.constants.promoted import (
PROMOTED_GROUP_CHOICES,
PROMOTED_GROUPS_BY_ID,
)
from olympia.constants.reviewers import REPUTATION_CHOICES
from olympia.files.models import File
from olympia.files.utils import extract_translations, resolve_i18n_message
Expand Down Expand Up @@ -1571,9 +1574,13 @@ def promoted_group(self, *, currently_approved=True):
try:
promoted = self.promotedaddon
except PromotedAddon.DoesNotExist:
return NOT_PROMOTED
return PROMOTED_GROUPS_BY_ID[PROMOTED_GROUP_CHOICES.NOT_PROMOTED]
is_promoted = not currently_approved or promoted.approved_applications
return promoted.group if is_promoted else NOT_PROMOTED
return (
promoted.group
if is_promoted
else PROMOTED_GROUPS_BY_ID[PROMOTED_GROUP_CHOICES.NOT_PROMOTED]
)

@cached_property
def promoted(self):
Expand All @@ -1584,7 +1591,9 @@ def promoted(self):
from olympia.promoted.models import PromotedTheme

if self._is_recommended_theme():
return PromotedTheme(addon=self, group_id=RECOMMENDED.id)
return PromotedTheme(
addon=self, group_id=PROMOTED_GROUP_CHOICES.RECOMMENDED
)
return None

@cached_property
Expand Down
12 changes: 8 additions & 4 deletions src/olympia/addons/serializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@
from olympia.constants.applications import APP_IDS, APPS_ALL
from olympia.constants.base import ADDON_TYPE_CHOICES_API
from olympia.constants.categories import CATEGORIES_BY_ID
from olympia.constants.promoted import PROMOTED_GROUPS, RECOMMENDED
from olympia.constants.promoted import PROMOTED_GROUP_CHOICES
from olympia.core.languages import AMO_LANGUAGES
from olympia.files.models import File, FileUpload
from olympia.files.utils import DuplicateAddonID, parse_addon
Expand Down Expand Up @@ -962,9 +962,11 @@ def validate_user_id(self, value):


class PromotedAddonSerializer(AMOModelSerializer):
GROUP_CHOICES = [(group.id, group.api_name) for group in PROMOTED_GROUPS]
apps = serializers.SerializerMethodField()
category = ReverseChoiceField(choices=GROUP_CHOICES, source='group_id')
category = ReverseChoiceField(
choices=PROMOTED_GROUP_CHOICES.api_choices,
source='group_id',
)

class Meta:
model = PromotedAddon
Expand Down Expand Up @@ -1144,7 +1146,9 @@ def get_has_eula(self, obj):
def get_is_featured(self, obj):
# featured is gone, but we need to keep the API backwards compatible so
# fake it with promoted status instead.
return bool(obj.promoted and obj.promoted.group == RECOMMENDED)
return bool(
obj.promoted and obj.promoted.group_id == PROMOTED_GROUP_CHOICES.RECOMMENDED
)

def get_has_privacy_policy(self, obj):
return bool(getattr(obj, 'has_privacy_policy', obj.privacy_policy))
Expand Down
8 changes: 4 additions & 4 deletions src/olympia/addons/tests/test_indexers.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
from olympia.bandwagon.models import Collection
from olympia.constants.applications import FIREFOX
from olympia.constants.licenses import LICENSES_BY_BUILTIN
from olympia.constants.promoted import RECOMMENDED
from olympia.constants.promoted import PROMOTED_GROUP_CHOICES
from olympia.constants.search import SEARCH_LANGUAGE_TO_ANALYZER
from olympia.files.models import WebextPermission
from olympia.versions.compare import version_int
Expand Down Expand Up @@ -512,10 +512,10 @@ def test_extract_promoted(self):
assert extracted['is_recommended'] is False

# Promoted extension.
self.addon = addon_factory(promoted=RECOMMENDED)
self.addon = addon_factory(promoted_id=PROMOTED_GROUP_CHOICES.RECOMMENDED)
extracted = self._extract()
assert extracted['promoted']
assert extracted['promoted']['group_id'] == RECOMMENDED.id
assert extracted['promoted']['group_id'] == PROMOTED_GROUP_CHOICES.RECOMMENDED
assert extracted['promoted']['approved_for_apps'] == [
amo.FIREFOX.id,
amo.ANDROID.id,
Expand All @@ -536,7 +536,7 @@ def test_extract_promoted(self):
featured_collection.add_addon(self.addon)
extracted = self._extract()
assert extracted['promoted']
assert extracted['promoted']['group_id'] == RECOMMENDED.id
assert extracted['promoted']['group_id'] == PROMOTED_GROUP_CHOICES.RECOMMENDED
assert extracted['promoted']['approved_for_apps'] == [
amo.FIREFOX.id,
amo.ANDROID.id,
Expand Down
39 changes: 24 additions & 15 deletions src/olympia/addons/tests/test_models.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@
from olympia.bandwagon.models import Collection
from olympia.blocklist.models import Block, BlocklistSubmission
from olympia.constants.categories import CATEGORIES
from olympia.constants.promoted import LINE, NOT_PROMOTED, RECOMMENDED, SPOTLIGHT
from olympia.constants.promoted import PROMOTED_GROUP_CHOICES
from olympia.devhub.models import RssKey
from olympia.files.models import File
from olympia.files.tests.test_models import UploadMixin
Expand Down Expand Up @@ -1640,33 +1640,38 @@ def test_promoted_group(self):
# default case - no group so not recommended
assert not addon.promoted_group()
# NOT_PROMOTED is falsey
assert addon.promoted_group() == NOT_PROMOTED
assert addon.promoted_group().id == PROMOTED_GROUP_CHOICES.NOT_PROMOTED
assert not addon.promoted_group(currently_approved=False)

# It's promoted but nothing has been approved
promoted = PromotedAddon.objects.create(addon=addon, group_id=LINE.id)
promoted = PromotedAddon.objects.create(
addon=addon, group_id=PROMOTED_GROUP_CHOICES.LINE
)
assert addon.promoted_group(currently_approved=False)
assert addon.promoted_group() == NOT_PROMOTED
assert addon.promoted_group().id == PROMOTED_GROUP_CHOICES.NOT_PROMOTED
assert not addon.promoted_group()

# The latest version is approved for the same group.
promoted.approve_for_version(version=addon.current_version)
assert addon.promoted_group()
assert addon.promoted_group() == LINE
assert addon.promoted_group().id == PROMOTED_GROUP_CHOICES.LINE

# if the group has changes the approval for the current version isn't
# valid
promoted.update(group_id=SPOTLIGHT.id)
promoted.update(group_id=PROMOTED_GROUP_CHOICES.SPOTLIGHT)
assert not addon.promoted_group()
assert addon.promoted_group(currently_approved=False)
assert addon.promoted_group(currently_approved=False) == SPOTLIGHT
assert (
addon.promoted_group(currently_approved=False).id
== PROMOTED_GROUP_CHOICES.SPOTLIGHT
)

promoted.approve_for_version(version=addon.current_version)
assert addon.promoted_group() == SPOTLIGHT
assert addon.promoted_group().id == PROMOTED_GROUP_CHOICES.SPOTLIGHT

# Application specific group membership should work too
# if no app is specifed in the PromotedAddon everything should match
assert addon.promoted_group() == SPOTLIGHT
assert addon.promoted_group().id == PROMOTED_GROUP_CHOICES.SPOTLIGHT
# update to mobile app
promoted.update(application_id=amo.ANDROID.id)
assert addon.promoted_group()
Expand All @@ -1677,14 +1682,14 @@ def test_promoted_group(self):
del addon.current_version.approved_for_groups
assert not addon.promoted_group()
promoted.update(application_id=amo.FIREFOX.id)
assert addon.promoted_group() == SPOTLIGHT
assert addon.promoted_group().id == PROMOTED_GROUP_CHOICES.SPOTLIGHT

# check it doesn't error if there's no current_version
addon.current_version.file.update(status=amo.STATUS_DISABLED)
addon.update_version()
assert not addon.current_version
assert not addon.promoted_group()
assert addon.promoted_group() == NOT_PROMOTED
assert addon.promoted_group().id == PROMOTED_GROUP_CHOICES.NOT_PROMOTED
assert addon.promoted_group(currently_approved=False)

def test_promoted(self):
Expand All @@ -1693,7 +1698,9 @@ def test_promoted(self):
assert addon.promoted is None

# It's promoted but nothing has been approved.
promoted = PromotedAddon.objects.create(addon=addon, group_id=LINE.id)
promoted = PromotedAddon.objects.create(
addon=addon, group_id=PROMOTED_GROUP_CHOICES.LINE
)
assert addon.promoted is None

# The latest version is approved.
Expand All @@ -1703,7 +1710,7 @@ def test_promoted(self):

# If the group changes the approval for the current version isn't
# valid.
promoted.update(group_id=SPOTLIGHT.id)
promoted.update(group_id=PROMOTED_GROUP_CHOICES.SPOTLIGHT)
del addon.promoted
assert addon.promoted is None

Expand All @@ -1725,7 +1732,7 @@ def test_promoted_theme(self):
# it's in the collection, so is now promoted.
assert addon.promoted
assert addon.promoted.addon == addon
assert addon.promoted.group_id == RECOMMENDED.id
assert addon.promoted.group_id == PROMOTED_GROUP_CHOICES.RECOMMENDED
assert addon.promoted.application_id is None
# This PromotedAddon instance is not a saved one.
assert addon.promoted.id is None
Expand Down Expand Up @@ -1776,7 +1783,9 @@ def test_can_be_compatible_with_all_fenix_versions_property(self):
assert not addon.can_be_compatible_with_all_fenix_versions

# It's promoted but nothing has been approved.
promoted = PromotedAddon.objects.create(addon=addon, group_id=LINE.id)
promoted = PromotedAddon.objects.create(
addon=addon, group_id=PROMOTED_GROUP_CHOICES.LINE
)
assert not addon.can_be_compatible_with_all_fenix_versions

# The latest version is approved.
Expand Down
Loading
Loading