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

PORT-5598 | Jira integration - support sprints | Change priority to name #804

Closed
wants to merge 118 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
118 commits
Select commit Hold shift + click to select a range
a3d6380
Implemeneted support for sprints
lordsarcastic Jul 10, 2024
2a2811c
Changed priority in issue to be name
lordsarcastic Jul 10, 2024
2d3f527
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Jul 10, 2024
7121711
Implemented support for sprint states and issues source
lordsarcastic Jul 16, 2024
be33aef
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Jul 16, 2024
0367e78
Fix type cast attribute overlap error
lordsarcastic Jul 16, 2024
9307d15
Formatted port-app-config
lordsarcastic Jul 16, 2024
b9abc49
Added new mapping for board url
lordsarcastic Jul 18, 2024
75a090a
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Jul 18, 2024
67966a1
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Jul 23, 2024
6b6cb82
Fixed issues on jira integration
lordsarcastic Jul 26, 2024
58f9984
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Jul 26, 2024
1fe260f
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Jul 31, 2024
ec6c1a0
Implemented processing for all webhook events
lordsarcastic Aug 1, 2024
320caa2
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Aug 1, 2024
32d82d3
Fix unclosed pagination for sprint
lordsarcastic Aug 1, 2024
1a6a376
Fix bugs in webhook
lordsarcastic Aug 1, 2024
90f4071
Fixed bug in deleting entities
lordsarcastic Aug 2, 2024
db696f1
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Aug 6, 2024
5df8c3d
Merge branch 'main' into port-5598
mk-armah Aug 9, 2024
ff31969
Fix Nonetype error
lordsarcastic Aug 14, 2024
41e7fcc
Bumped application version
lordsarcastic Aug 14, 2024
a447370
Update changelog
lordsarcastic Aug 14, 2024
69458cc
Merge branch 'port-5598' of github.com:port-labs/ocean into port-5598
lordsarcastic Aug 19, 2024
91b406d
Remove sourcing issues from boards
lordsarcastic Aug 19, 2024
4362377
Ran formatting
lordsarcastic Aug 19, 2024
df881df
Chore: Removed boards entirely
lordsarcastic Aug 26, 2024
88899ba
Implemeneted support for sprints
lordsarcastic Jul 10, 2024
ac6d419
Changed priority in issue to be name
lordsarcastic Jul 10, 2024
0be072f
Implemented support for sprint states and issues source
lordsarcastic Jul 16, 2024
d9d4f72
Fix type cast attribute overlap error
lordsarcastic Jul 16, 2024
8560691
Formatted port-app-config
lordsarcastic Jul 16, 2024
b19e8b7
Added new mapping for board url
lordsarcastic Jul 18, 2024
d4b0242
Fixed issues on jira integration
lordsarcastic Jul 26, 2024
a616d5b
Implemented processing for all webhook events
lordsarcastic Aug 1, 2024
f72fd3e
Fix unclosed pagination for sprint
lordsarcastic Aug 1, 2024
1158543
Fix bugs in webhook
lordsarcastic Aug 1, 2024
d2fa837
Fixed bug in deleting entities
lordsarcastic Aug 2, 2024
d761c9e
[Integration][Pagerduty] Fix incident default mapping and blueprint …
Tankilevitch Aug 5, 2024
fd9cf23
PORT-9572 Update the default JQL provided by the JIRA integration (#868)
MPTG94 Aug 5, 2024
e97e3ab
[AWS] Bug fix global resources not synced for all accounts (#845)
shalev007 Aug 5, 2024
59cdfa6
[Integration][AWS] throw error if no permissions on region (#889)
shalev007 Aug 8, 2024
012474c
Remove sourcing issues from boards
lordsarcastic Aug 19, 2024
b755efa
Ran formatting
lordsarcastic Aug 19, 2024
52cd217
Chore: Removed boards entirely
lordsarcastic Aug 26, 2024
6a5a123
Merge branch 'port-5598' of github.com:port-labs/ocean into port-5598
lordsarcastic Aug 26, 2024
fd1731d
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Aug 26, 2024
f14d717
Merge branch 'main' into port-5598
PeyGis Sep 4, 2024
6aa8f99
Fix: Fixed ingesting issues with sprint
lordsarcastic Sep 5, 2024
028eb1a
Fix: Single quote causing mapping to not be applied
lordsarcastic Sep 5, 2024
266256a
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Sep 11, 2024
5eb0eb2
Chore: Made ingestion faster
lordsarcastic Sep 11, 2024
04de891
Fix: Lint issues
lordsarcastic Sep 11, 2024
6742a22
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Sep 13, 2024
4423064
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Sep 19, 2024
c9e5345
Added changelog information
lordsarcastic Sep 19, 2024
4fca62a
Bumped version
lordsarcastic Sep 19, 2024
264faa7
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Sep 25, 2024
64ecbd1
Chore: bumped version
lordsarcastic Sep 25, 2024
ebe15f9
Chore: Removed sprints from integration
lordsarcastic Oct 4, 2024
c74edb7
Fix: Bug on params for issues
lordsarcastic Oct 7, 2024
444d177
Fix: Bug on params for issues
lordsarcastic Oct 8, 2024
80b388b
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Oct 10, 2024
a572710
Merge branch 'main' into port-5598
PeyGis Oct 14, 2024
d4f36f6
Fix: Changelog, blueprints and method name
lordsarcastic Oct 14, 2024
4f8ca50
Merge branch 'main' into port-5598
PeyGis Oct 14, 2024
4602bd1
Implemented tests for Jira
lordsarcastic Oct 17, 2024
bb26949
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Oct 18, 2024
1eaf4fb
Merge branch 'main' into port-5598
PeyGis Oct 21, 2024
bd6c269
Fixed bug in tests
lordsarcastic Oct 21, 2024
8b6acc5
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Oct 22, 2024
ec3727f
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Oct 22, 2024
8c9c29a
Bumped integration version and ocean dependency
lordsarcastic Oct 22, 2024
34acfc9
Updated changelog
lordsarcastic Oct 22, 2024
8e34aa1
Fix: Lint errors
lordsarcastic Oct 22, 2024
19f49bb
Fix: Uncomment items in spec.yaml file
lordsarcastic Oct 22, 2024
1879b0b
Fix: commented code
lordsarcastic Oct 22, 2024
52d6a8c
Chore: Adjusted fixtures data
lordsarcastic Oct 22, 2024
bdf93d1
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Oct 23, 2024
f1f8495
Bumped version
lordsarcastic Oct 23, 2024
f6668e6
Chore: Fix lint and lock file
lordsarcastic Oct 23, 2024
bae8656
Chore: Reverted changelog changes
lordsarcastic Oct 23, 2024
8d663f6
Chore: Updated changelog
lordsarcastic Oct 23, 2024
9ae1adc
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Oct 23, 2024
4dd0f85
Bumped integration version due to merge
lordsarcastic Oct 23, 2024
0886010
Chore: Fixed changelog
lordsarcastic Oct 25, 2024
303803a
Chore: Implemented tests for client
lordsarcastic Nov 1, 2024
f5a3e0f
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Nov 1, 2024
dd27cab
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Nov 4, 2024
c01eb08
Fixed all failing tests
lordsarcastic Nov 5, 2024
c6ac6b2
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Nov 5, 2024
3bf1dc2
Fixed failing CI test
lordsarcastic Nov 7, 2024
c2388f3
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Nov 7, 2024
72862ad
Bumped integration version
lordsarcastic Nov 7, 2024
0fac699
Merge branch 'main' into port-5598
Tankilevitch Nov 7, 2024
e572c33
Chore: Made event handling robust
lordsarcastic Nov 7, 2024
9fb5c95
Merge branch 'port-5598' of github.com:port-labs/ocean into port-5598
lordsarcastic Nov 7, 2024
e3da008
Removed useless logs
lordsarcastic Nov 7, 2024
8418bc5
Chore: Added test cases for webhooks
lordsarcastic Nov 8, 2024
d4f36ee
Merge branch 'main' of github.com:port-labs/Port-Ocean into port-5598
Tankilevitch Nov 10, 2024
636e62b
Merge branch 'port-5598' of github.com:port-labs/Port-Ocean into port…
Tankilevitch Nov 10, 2024
a479518
Chore: Commented out webhook tests
lordsarcastic Nov 11, 2024
48d3ba3
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Nov 11, 2024
e898b13
buggy: testing issue selector equality
lordsarcastic Nov 12, 2024
c018609
fix instance not matching in resource config
mk-armah Nov 12, 2024
cfb22ab
Fixed webhooks in jira integratin
lordsarcastic Nov 12, 2024
65cfce3
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Nov 12, 2024
bccfa16
Chore: Bumped integration version
lordsarcastic Nov 12, 2024
0766f28
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Nov 13, 2024
0351626
Chore: Added mock ocean context fixture to full sync test
lordsarcastic Nov 14, 2024
c0626b4
[DROP] Erik test pytest concurrency
erikzaadi Nov 27, 2024
6b3784a
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Nov 28, 2024
4c2613c
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Dec 5, 2024
6b71420
Chore: resolved conflicts
lordsarcastic Dec 5, 2024
ffad733
Chore: commented out tests
lordsarcastic Dec 5, 2024
9db53c2
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Dec 9, 2024
f25dc6c
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Dec 9, 2024
a6f99c1
Merge remote-tracking branch 'origin/main' into port-5598
lordsarcastic Dec 10, 2024
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
2 changes: 1 addition & 1 deletion .github/workflows/integrations-test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ jobs:
- name: Test
working-directory: ${{ format('integrations/{0}', matrix.folder) }}
env:
PYTEST_ADDOPTS: --junitxml=junit/test-results-${{ format('integrations/{0}', matrix.folder) }}.xml
PYTEST_ADDOPTS: --junitxml=junit/test-results-${{ format('integrations/{0}', matrix.folder) }}.xml --dist=no
run: |
make test

Expand Down
9 changes: 8 additions & 1 deletion integrations/jira/.port/resources/blueprints.json
Original file line number Diff line number Diff line change
Expand Up @@ -11,10 +11,17 @@
"type": "string",
"format": "url",
"description": "URL to the project in Jira"
},
"totalIssues": {
"title": "Total Issues",
"type": "number",
"description": "The total number of issues in the project"
}
}
},
"calculationProperties": {}
"mirrorProperties": {},
"calculationProperties": {},
"relations": {}
},
{
"identifier": "jiraUser",
Expand Down
4 changes: 3 additions & 1 deletion integrations/jira/.port/resources/port-app-config.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@ resources:
blueprint: '"jiraProject"'
properties:
url: (.self | split("/") | .[:3] | join("/")) + "/projects/" + .key
totalIssues: .insight.totalIssueCount


- kind: user
selector:
Expand Down Expand Up @@ -47,7 +49,7 @@ resources:
issueType: .fields.issuetype.name
components: .fields.components
creator: .fields.creator.emailAddress
priority: .fields.priority.id
priority: .fields.priority.name
labels: .fields.labels
created: .fields.created
updated: .fields.updated
Expand Down
13 changes: 13 additions & 0 deletions integrations/jira/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,19 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Bumped ocean version to ^0.14.5


## 0.1.104 (2024-11-27)


### Features

- Added a field to display total issues in a project
- Added support for ingesting other fields apart from the default fields (Jira Sprint support)

### Improvements

- Changed issue priority from id to name


## 0.1.103 (2024-11-25)


Expand Down
183 changes: 183 additions & 0 deletions integrations/jira/client.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,183 @@
import typing
from typing import Any, AsyncGenerator, Generator

import httpx
from httpx import Auth, BasicAuth, Request, Response, Timeout
from loguru import logger
from port_ocean.context.ocean import ocean
from port_ocean.utils import http_async_client

PAGE_SIZE = 50
WEBHOOK_NAME = "Port-Ocean-Events-Webhook"
REQUEST_TIMEOUT = 120


CREATE_UPDATE_WEBHOOK_EVENTS = [
"jira:issue_created",
"jira:issue_updated",
"project_created",
"project_updated",
"project_restored_deleted",
"project_restored_archived",
"user_created",
"user_updated",
]

DELETE_WEBHOOK_EVENTS = [
"jira:issue_deleted",
"project_deleted",
"project_soft_deleted",
"project_archived",
"user_deleted",
]

WEBHOOK_EVENTS = [
*CREATE_UPDATE_WEBHOOK_EVENTS,
*DELETE_WEBHOOK_EVENTS,
]


class BearerAuth(Auth):
def __init__(self, token: str):
self.token = token

def auth_flow(self, request: Request) -> Generator[Request, Response, None]:
request.headers["Authorization"] = f"Bearer {self.token}"
yield request


class JiraClient:
def __init__(self, jira_url: str, jira_email: str, jira_token: str) -> None:
self.jira_url = jira_url
self.agile_url = f"{self.jira_url}/rest/agile/1.0"
self.jira_rest_url = f"{self.jira_url}/rest"
self.detail_base_url = f"{self.jira_rest_url}/api/3"

# If the Jira URL is directing to api.atlassian.com, we use OAuth2 Bearer Auth
self.jira_api_auth: Auth
if "api.atlassian.com" in self.jira_url:
self.jira_api_auth = BearerAuth(jira_token)
else:
self.jira_api_auth = BasicAuth(jira_email, jira_token)
self.webhooks_url = f"{self.jira_rest_url}/webhooks/1.0/webhook"

self.client = http_async_client
self.client.auth = self.jira_api_auth
self.client.timeout = Timeout(REQUEST_TIMEOUT)

@staticmethod
def _generate_base_req_params(
maxResults: int = 50, startAt: int = 0
) -> dict[str, Any]:
return {
"maxResults": maxResults,
"startAt": startAt,
}

async def _make_paginated_request(
self,
url: str,
params: dict[str, Any] = {},
is_last_function: typing.Callable[
[dict[str, Any]], bool
] = lambda response: response["isLast"],
) -> AsyncGenerator[dict[str, list[dict[str, Any]]], None]:
params = {**self._generate_base_req_params(), **params}
is_last = False
logger.info(f"Making paginated request to {url} with params: {params}")
while not is_last:
try:
response = await self.client.get(url, params=params)
response.raise_for_status()
response_data = response.json()
yield response_data
is_last = is_last_function(response_data)
start = response_data["startAt"] + response_data["maxResults"]
params = {**params, "startAt": start}
logger.info(f"Next page startAt: {start}")
except httpx.HTTPStatusError as e:

logger.error(
f"HTTP error with status code: {e.response.status_code}"
f" and response text: {e.response.text}"
)
raise
except httpx.HTTPError as e:
logger.error(f"HTTP occurred while fetching Jira data {e}")
raise
logger.info("Finished paginated request")
return

async def get_all_projects(self) -> AsyncGenerator[list[dict[str, Any]], None]:
async for projects in self._make_paginated_request(
f"{self.detail_base_url}/project/search"
):
yield projects["values"]

async def get_all_issues(
self,
params: dict[str, Any] = {},
) -> AsyncGenerator[list[dict[str, Any]], None]:
async for issues in self._make_paginated_request(
f"{self.detail_base_url}/search",
params=params,
is_last_function=lambda response: response["startAt"]
+ response["maxResults"]
>= response["total"],
):
yield issues["issues"]

async def get_all_users(self, params: dict[str, Any] = {}) -> list[dict[str, Any]]:
return await self._get_single_item(
f"{self.detail_base_url}/users", params=params
)

async def _get_single_item(self, url: str, params: dict[str, Any] = {}) -> Any:
try:
response = await self.client.get(url, params=params)
response.raise_for_status()
return response.json()
except httpx.HTTPStatusError as e:
logger.error(
f"HTTP error on {url}: {e.response.status_code} - {e.response.text}"
)
raise
except httpx.HTTPError as e:
logger.error(f"HTTP occurred while fetching Jira data {e}")
raise

async def get_single_project(self, project: str) -> dict[str, Any]:
return await self._get_single_item(f"{self.detail_base_url}/project/{project}")

async def get_single_issue(
self, issue: str, fields: dict[str, Any] = {}, jql: str | None = None
) -> dict[str, Any]:
return await self._get_single_item(f"{self.agile_url}/issue/{issue}")

async def get_single_user(self, account_id: str) -> dict[str, Any]:
return await self._get_single_item(
f"{self.detail_base_url}/user", params={"accountId": account_id}
)

async def create_events_webhook(self, app_host: str) -> None:
webhook_target_app_host = f"{app_host}/integration/webhook"
webhook_check_response = await self.client.get(f"{self.webhooks_url}")
webhook_check_response.raise_for_status()
webhook_check = webhook_check_response.json()

for webhook in webhook_check:
if webhook["url"] == webhook_target_app_host:
logger.info("Ocean real time reporting webhook already exists")
return

body = {
"name": f"{ocean.config.integration.identifier}-{WEBHOOK_NAME}",
"url": webhook_target_app_host,
"events": WEBHOOK_EVENTS,
}

webhook_create_response = await self.client.post(
f"{self.webhooks_url}", json=body
)
webhook_create_response.raise_for_status()
logger.info("Ocean real time reporting webhook created")
2 changes: 1 addition & 1 deletion integrations/jira/integration.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
from port_ocean.core.handlers.port_app_config.api import APIPortAppConfig
from port_ocean.core.integrations.base import BaseIntegration

from jira.overrides import JiraPortAppConfig
from overrides import JiraPortAppConfig


class JiraIntegration(BaseIntegration):
Expand Down
Empty file.
Loading
Loading