Skip to content

Commit 8d317ae

Browse files
authored
add signal for Owner (#940)
1 parent 897327a commit 8d317ae

File tree

9 files changed

+252
-143
lines changed

9 files changed

+252
-143
lines changed

codecov/settings_test.py

+8-18
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,12 @@
1-
from .settings_base import *
1+
import os
2+
3+
from .settings_dev import *
24

3-
DEBUG = False
45
ALLOWED_HOSTS = ["localhost"]
5-
WEBHOOK_URL = "" # NGROK TUNNEL HERE
6-
STRIPE_API_KEY = ""
76
CORS_ALLOWED_ORIGINS = ["http://localhost:9000", "http://localhost"]
8-
CORS_ALLOW_CREDENTIALS = True
9-
CODECOV_URL = "localhost"
10-
CODECOV_API_URL = get_config("setup", "codecov_api_url", default=CODECOV_URL)
11-
DATABASE_HOST = "postgres"
7+
SHELTER_PUBSUB_PROJECT_ID = "test-project-id"
8+
SHELTER_PUBSUB_SYNC_REPO_TOPIC_ID = "test-topic-id"
129

13-
DATABASES = {
14-
"default": {
15-
"ENGINE": "django.db.backends.postgresql",
16-
"NAME": DATABASE_NAME,
17-
"USER": DATABASE_USER,
18-
"PASSWORD": DATABASE_PASSWORD,
19-
"HOST": DATABASE_HOST,
20-
"PORT": "5432",
21-
}
22-
}
10+
# Mock the Pub/Sub host for testing
11+
# this prevents the pubsub SDK from trying to load credentials
12+
os.environ["PUBSUB_EMULATOR_HOST"] = "localhost"

codecov_auth/signals.py

+37-31
Original file line numberDiff line numberDiff line change
@@ -1,47 +1,53 @@
1-
import json
1+
from typing import Any, Dict, Optional, Type
22

3-
from django.conf import settings
43
from django.db.models.signals import post_save
54
from django.dispatch import receiver
6-
from google.cloud import pubsub_v1
75

86
from codecov_auth.models import OrganizationLevelToken, Owner, OwnerProfile
7+
from utils.shelter import ShelterPubsub
98

109

1110
@receiver(post_save, sender=Owner)
1211
def create_owner_profile_when_owner_is_created(
13-
sender, instance: Owner, created, **kwargs
14-
):
12+
sender: Type[Owner], instance: Owner, created: bool, **kwargs: Dict[str, Any]
13+
) -> Optional[OwnerProfile]:
1514
if created:
1615
return OwnerProfile.objects.create(owner_id=instance.ownerid)
1716

1817

19-
_pubsub_publisher = None
20-
21-
22-
def _get_pubsub_publisher():
23-
global _pubsub_publisher
24-
if not _pubsub_publisher:
25-
_pubsub_publisher = pubsub_v1.PublisherClient()
26-
return _pubsub_publisher
27-
28-
2918
@receiver(
3019
post_save, sender=OrganizationLevelToken, dispatch_uid="shelter_sync_org_token"
3120
)
32-
def update_org_token(sender, instance: OrganizationLevelToken, **kwargs):
33-
pubsub_project_id = settings.SHELTER_PUBSUB_PROJECT_ID
34-
topic_id = settings.SHELTER_PUBSUB_SYNC_REPO_TOPIC_ID
35-
if pubsub_project_id and topic_id:
36-
publisher = _get_pubsub_publisher()
37-
topic_path = publisher.topic_path(pubsub_project_id, topic_id)
38-
publisher.publish(
39-
topic_path,
40-
json.dumps(
41-
{
42-
"type": "org_token",
43-
"sync": "one",
44-
"id": instance.id,
45-
}
46-
).encode("utf-8"),
47-
)
21+
def update_org_token(
22+
sender: Type[OrganizationLevelToken],
23+
instance: OrganizationLevelToken,
24+
**kwargs: Dict[str, Any],
25+
) -> None:
26+
data = {
27+
"type": "org_token",
28+
"sync": "one",
29+
"id": instance.id,
30+
}
31+
ShelterPubsub.get_instance().publish(data)
32+
33+
34+
@receiver(post_save, sender=Owner, dispatch_uid="shelter_sync_owner")
35+
def update_owner(
36+
sender: Type[Owner], instance: Owner, **kwargs: Dict[str, Any]
37+
) -> None:
38+
"""
39+
Shelter tracks a limited set of Owner fields - only update if those fields have changed.
40+
"""
41+
created: bool = kwargs["created"]
42+
tracked_fields = [
43+
"upload_token_required_for_public_repos",
44+
"username",
45+
"service",
46+
]
47+
if created or any(instance.tracker.has_changed(field) for field in tracked_fields):
48+
data = {
49+
"type": "owner",
50+
"sync": "one",
51+
"id": instance.ownerid,
52+
}
53+
ShelterPubsub.get_instance().publish(data)

codecov_auth/tests/test_signals.py

+107-13
Original file line numberDiff line numberDiff line change
@@ -1,27 +1,121 @@
1-
import os
1+
from unittest import mock
2+
from unittest.mock import call
23

34
import pytest
4-
from django.test import override_settings
5+
from django.test import TestCase
6+
from shared.django_apps.codecov_auth.models import Service
57
from shared.django_apps.codecov_auth.tests.factories import (
68
OrganizationLevelTokenFactory,
9+
OwnerFactory,
710
)
811

912

10-
@override_settings(
11-
SHELTER_PUBSUB_PROJECT_ID="test-project-id",
12-
SHELTER_PUBSUB_SYNC_REPO_TOPIC_ID="test-topic-id",
13-
)
1413
@pytest.mark.django_db
1514
def test_shelter_org_token_sync(mocker):
16-
# this prevents the pubsub SDK from trying to load credentials
17-
os.environ["PUBSUB_EMULATOR_HOST"] = "localhost"
18-
1915
publish = mocker.patch("google.cloud.pubsub_v1.PublisherClient.publish")
2016

2117
# this triggers the publish via Django signals
22-
OrganizationLevelTokenFactory(id=91728376)
18+
OrganizationLevelTokenFactory(id=91728376, owner=OwnerFactory(ownerid=111))
2319

24-
publish.assert_called_once_with(
25-
"projects/test-project-id/topics/test-topic-id",
26-
b'{"type": "org_token", "sync": "one", "id": 91728376}',
20+
publish.assert_has_calls(
21+
[
22+
call(
23+
"projects/test-project-id/topics/test-topic-id",
24+
b'{"type": "owner", "sync": "one", "id": 111}',
25+
),
26+
call(
27+
"projects/test-project-id/topics/test-topic-id",
28+
b'{"type": "org_token", "sync": "one", "id": 91728376}',
29+
),
30+
]
2731
)
32+
33+
34+
@mock.patch("google.cloud.pubsub_v1.PublisherClient.publish")
35+
class TestCodecovAuthSignals(TestCase):
36+
def test_sync_on_create(self, mock_publish):
37+
OwnerFactory(ownerid=12345)
38+
mock_publish.assert_called_once_with(
39+
"projects/test-project-id/topics/test-topic-id",
40+
b'{"type": "owner", "sync": "one", "id": 12345}',
41+
)
42+
43+
def test_sync_on_update_upload_token_required_for_public_repos(self, mock_publish):
44+
owner = OwnerFactory(ownerid=12345, upload_token_required_for_public_repos=True)
45+
owner.upload_token_required_for_public_repos = False
46+
owner.save()
47+
mock_publish.assert_has_calls(
48+
[
49+
call(
50+
"projects/test-project-id/topics/test-topic-id",
51+
b'{"type": "owner", "sync": "one", "id": 12345}',
52+
),
53+
call(
54+
"projects/test-project-id/topics/test-topic-id",
55+
b'{"type": "owner", "sync": "one", "id": 12345}',
56+
),
57+
]
58+
)
59+
60+
def test_sync_on_update_username(self, mock_publish):
61+
owner = OwnerFactory(ownerid=12345, username="hello")
62+
owner.username = "world"
63+
owner.save()
64+
mock_publish.assert_has_calls(
65+
[
66+
call(
67+
"projects/test-project-id/topics/test-topic-id",
68+
b'{"type": "owner", "sync": "one", "id": 12345}',
69+
),
70+
call(
71+
"projects/test-project-id/topics/test-topic-id",
72+
b'{"type": "owner", "sync": "one", "id": 12345}',
73+
),
74+
]
75+
)
76+
77+
def test_sync_on_update_service(self, mock_publish):
78+
owner = OwnerFactory(ownerid=12345, service=Service.GITHUB.value)
79+
owner.service = Service.BITBUCKET.value
80+
owner.save()
81+
mock_publish.assert_has_calls(
82+
[
83+
call(
84+
"projects/test-project-id/topics/test-topic-id",
85+
b'{"type": "owner", "sync": "one", "id": 12345}',
86+
),
87+
call(
88+
"projects/test-project-id/topics/test-topic-id",
89+
b'{"type": "owner", "sync": "one", "id": 12345}',
90+
),
91+
]
92+
)
93+
94+
def test_no_sync_on_update_other_fields(self, mock_publish):
95+
owner = OwnerFactory(ownerid=12345, name="hello")
96+
owner.name = "world"
97+
owner.save()
98+
mock_publish.assert_called_once_with(
99+
"projects/test-project-id/topics/test-topic-id",
100+
b'{"type": "owner", "sync": "one", "id": 12345}',
101+
)
102+
103+
@mock.patch("logging.Logger.warning")
104+
def test_sync_error(self, mock_log, mock_publish):
105+
mock_publish.side_effect = Exception("publish error")
106+
107+
OwnerFactory(ownerid=12345)
108+
109+
# publish is still called, raises an Exception
110+
mock_publish.assert_called_once_with(
111+
"projects/test-project-id/topics/test-topic-id",
112+
b'{"type": "owner", "sync": "one", "id": 12345}',
113+
)
114+
115+
mock_log.assert_called_once_with(
116+
"Failed to publish a message",
117+
extra=dict(
118+
data_to_publish={"type": "owner", "sync": "one", "id": 12345},
119+
error=mock_publish.side_effect,
120+
),
121+
)

core/signals.py

+26-51
Original file line numberDiff line numberDiff line change
@@ -1,68 +1,43 @@
1-
import json
21
import logging
2+
from typing import Any, Dict, List, Type
33

4-
from django.conf import settings
54
from django.db.models.signals import post_save
65
from django.dispatch import receiver
7-
from google.cloud import pubsub_v1
86
from shared.django_apps.core.models import Commit
97

108
from core.models import Repository
9+
from utils.shelter import ShelterPubsub
1110

12-
_pubsub_publisher = None
1311
log = logging.getLogger(__name__)
1412

1513

16-
def _get_pubsub_publisher():
17-
global _pubsub_publisher
18-
if not _pubsub_publisher:
19-
_pubsub_publisher = pubsub_v1.PublisherClient()
20-
return _pubsub_publisher
21-
22-
2314
@receiver(post_save, sender=Repository, dispatch_uid="shelter_sync_repo")
24-
def update_repository(sender, instance: Repository, **kwargs):
15+
def update_repository(
16+
sender: Type[Repository], instance: Repository, **kwargs: Dict[str, Any]
17+
) -> None:
2518
log.info(f"Signal triggered for repository {instance.repoid}")
26-
created = kwargs["created"]
27-
changes = instance.tracker.changed()
28-
if created or any([field in changes for field in ["name", "upload_token"]]):
29-
try:
30-
pubsub_project_id = settings.SHELTER_PUBSUB_PROJECT_ID
31-
topic_id = settings.SHELTER_PUBSUB_SYNC_REPO_TOPIC_ID
32-
if pubsub_project_id and topic_id:
33-
publisher = _get_pubsub_publisher()
34-
topic_path = publisher.topic_path(pubsub_project_id, topic_id)
35-
publisher.publish(
36-
topic_path,
37-
json.dumps(
38-
{
39-
"type": "repo",
40-
"sync": "one",
41-
"id": instance.repoid,
42-
}
43-
).encode("utf-8"),
44-
)
45-
log.info(f"Message published for repository {instance.repoid}")
46-
except Exception as e:
47-
log.warning(f"Failed to publish message for repo {instance.repoid}: {e}")
19+
created: bool = kwargs["created"]
20+
changes: Dict[str, Any] = instance.tracker.changed()
21+
tracked_fields: List[str] = ["name", "upload_token"]
22+
23+
if created or any([field in changes for field in tracked_fields]):
24+
data = {
25+
"type": "repo",
26+
"sync": "one",
27+
"id": instance.repoid,
28+
}
29+
ShelterPubsub.get_instance().publish(data)
4830

4931

5032
@receiver(post_save, sender=Commit, dispatch_uid="shelter_sync_commit")
51-
def update_commit(sender, instance: Commit, **kwargs):
52-
branch = instance.branch
33+
def update_commit(
34+
sender: Type[Commit], instance: Commit, **kwargs: Dict[str, Any]
35+
) -> None:
36+
branch: str = instance.branch
5337
if branch and ":" in branch:
54-
pubsub_project_id = settings.SHELTER_PUBSUB_PROJECT_ID
55-
topic_id = settings.SHELTER_PUBSUB_SYNC_REPO_TOPIC_ID
56-
if pubsub_project_id and topic_id:
57-
publisher = _get_pubsub_publisher()
58-
topic_path = publisher.topic_path(pubsub_project_id, topic_id)
59-
publisher.publish(
60-
topic_path,
61-
json.dumps(
62-
{
63-
"type": "commit",
64-
"sync": "one",
65-
"id": instance.id,
66-
}
67-
).encode("utf-8"),
68-
)
38+
data = {
39+
"type": "commit",
40+
"sync": "one",
41+
"id": instance.id,
42+
}
43+
ShelterPubsub.get_instance().publish(data)

0 commit comments

Comments
 (0)