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

Add support for URL shortening on Kibana 7.16 #633

Merged
merged 3 commits into from
Dec 30, 2021
Merged
Show file tree
Hide file tree
Changes from 2 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
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@
- sphinx 4.3.1 to 4.3.2 - [#618](https://github.com/jertel/elastalert2/pull/618) - @nsano-rururu
- Remove unused parameter boto-profile - [#622](https://github.com/jertel/elastalert2/pull/622) - @nsano-rururu
- [Docs] Include Docker example; add additional FAQs - [#623](https://github.com/jertel/elastalert2/pull/623) - @nsano-rururu
- Add support for URL shortening with Kibana 7.16+ - [#633](https://github.com/jertel/elastalert2/pull/633) - @jertel

# 2.2.3

Expand Down
35 changes: 27 additions & 8 deletions elastalert/kibana_external_url_formatter.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import boto3
import uuid
jertel marked this conversation as resolved.
Show resolved Hide resolved
import os
from urllib.parse import parse_qsl, urlencode, urljoin, urlparse, urlsplit, urlunsplit

Expand Down Expand Up @@ -46,12 +47,18 @@ def format(self, relative_url: str) -> str:
class ShortKibanaExternalUrlFormatter(KibanaExternalUrlFormatter):
'''Formats external urls using the Kibana Shorten URL API'''

def __init__(self, base_url: str, auth: AuthBase, security_tenant: str) -> None:
def __init__(self, base_url: str, auth: AuthBase, security_tenant: str, new_shortener: bool) -> None:
self.auth = auth
self.security_tenant = security_tenant
self.goto_url = urljoin(base_url, 'goto/')
self.use_new_shortener = new_shortener

shorten_url = urljoin(base_url, 'api/shorten_url')
if self.use_new_shortener:
path = 'api/short_url'
else:
path = 'api/shorten_url'

shorten_url = urljoin(base_url, path)
if security_tenant:
shorten_url = append_security_tenant(shorten_url, security_tenant)
self.shorten_url = shorten_url
Expand All @@ -62,6 +69,13 @@ def format(self, relative_url: str) -> str:
if self.security_tenant:
long_url = append_security_tenant(long_url, self.security_tenant)

if self.use_new_shortener:
json = { 'locatorId': 'LEGACY_SHORT_URL_LOCATOR', 'params': { 'url': long_url } }
response_param = 'id'
else:
json = { 'url': long_url }
response_param = 'urlId'

try:
response = requests.post(
url=self.shorten_url,
Expand All @@ -70,16 +84,14 @@ def format(self, relative_url: str) -> str:
'kbn-xsrf': 'elastalert',
'osd-xsrf': 'elastalert'
},
json={
'url': long_url
}
json=json
)
response.raise_for_status()
except RequestException as e:
raise EAException("Failed to invoke Kibana Shorten URL API: %s" % e)

response_body = response.json()
url_id = response_body.get('urlId')
url_id = response_body.get(response_param)

goto_url = urljoin(self.goto_url, url_id)
if self.security_tenant:
Expand Down Expand Up @@ -121,18 +133,25 @@ def create_kibana_auth(kibana_url, rule) -> AuthBase:
# Unauthenticated
return None

def is_kibana_atleastsevensixteen(version: str):
"""
Returns True when the Kibana server version >= 7.16
"""
major, minor = list(map(int, version.split(".")[:2]))
return major > 7 or (major == 7 and minor >= 16)

def create_kibana_external_url_formatter(
rule,
shorten: bool,
security_tenant: str
security_tenant: str,
) -> KibanaExternalUrlFormatter:
'''Creates a Kibana external url formatter'''

base_url = rule.get('kibana_url')
new_shortener = is_kibana_atleastsevensixteen(rule.get('kibana_discover_version', '0.0'))

if shorten:
auth = create_kibana_auth(base_url, rule)
return ShortKibanaExternalUrlFormatter(base_url, auth, security_tenant)
return ShortKibanaExternalUrlFormatter(base_url, auth, security_tenant, new_shortener)

return AbsoluteKibanaExternalUrlFormatter(base_url, security_tenant)
132 changes: 132 additions & 0 deletions tests/kibana_external_url_formatter_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,30 @@ def raise_for_status(self):
return MockResponse(400)


def mock_7_16_kibana_shorten_url_api(*args, **kwargs):
class MockResponse:
def __init__(self, status_code):
self.status_code = status_code

def json(self):
return {
'id': 'a1f77a80-6847-11ec-9b91-e5d43d1e9ca2'
}

def raise_for_status(self):
if self.status_code == 400:
raise requests.exceptions.HTTPError()

json = kwargs['json']
params = json['params']
url = params['url']

if url.startswith('/app/'):
return MockResponse(200)
else:
return MockResponse(400)


class ShortenUrlTestCase:
def __init__(
self,
Expand Down Expand Up @@ -200,6 +224,113 @@ def test_short_kinbana_external_url_formatter(
base_url=test_case.base_url,
auth=test_case.authorization,
security_tenant=test_case.security_tenant,
new_shortener=False,
)

actualUrl = formatter.format(test_case.relative_url)
assert actualUrl == test_case.expected_url

mock_post.assert_called_once_with(**test_case.expected_api_request)


@mock.patch('requests.post', side_effect=mock_7_16_kibana_shorten_url_api)
@pytest.mark.parametrize("test_case", [

# Relative to kibana plugin
ShortenUrlTestCase(
base_url='http://elasticsearch.test.org/_plugin/kibana/',
relative_url='app/dev_tools#/console',
expected_api_request={
'url': 'http://elasticsearch.test.org/_plugin/kibana/api/short_url',
'auth': None,
'headers': {
'kbn-xsrf': 'elastalert',
'osd-xsrf': 'elastalert'
},
'json': {
'locatorId': 'LEGACY_SHORT_URL_LOCATOR',
'params': {
'url': '/app/dev_tools#/console'
}
}
},
expected_url='http://elasticsearch.test.org/_plugin/kibana/goto/a1f77a80-6847-11ec-9b91-e5d43d1e9ca2'
),

# Relative to root of dedicated Kibana domain
ShortenUrlTestCase(
base_url='http://kibana.test.org/',
relative_url='/app/dev_tools#/console',
expected_api_request={
'url': 'http://kibana.test.org/api/short_url',
'auth': None,
'headers': {
'kbn-xsrf': 'elastalert',
'osd-xsrf': 'elastalert'
},
'json': {
'locatorId': 'LEGACY_SHORT_URL_LOCATOR',
'params': {
'url': '/app/dev_tools#/console'
}
}
},
expected_url='http://kibana.test.org/goto/a1f77a80-6847-11ec-9b91-e5d43d1e9ca2'
),

# With authentication
ShortenUrlTestCase(
base_url='http://kibana.test.org/',
auth=HTTPBasicAuth('john', 'doe'),
relative_url='/app/dev_tools#/console',
expected_api_request={
'url': 'http://kibana.test.org/api/short_url',
'auth': HTTPBasicAuth('john', 'doe'),
'headers': {
'kbn-xsrf': 'elastalert',
'osd-xsrf': 'elastalert'
},
'json': {
'locatorId': 'LEGACY_SHORT_URL_LOCATOR',
'params': {
'url': '/app/dev_tools#/console'
}
}
},
expected_url='http://kibana.test.org/goto/a1f77a80-6847-11ec-9b91-e5d43d1e9ca2'
),

# With security tenant
ShortenUrlTestCase(
base_url='http://kibana.test.org/',
security_tenant='global',
relative_url='/app/dev_tools#/console',
expected_api_request={
'url': 'http://kibana.test.org/api/short_url?security_tenant=global',
'auth': None,
'headers': {
'kbn-xsrf': 'elastalert',
'osd-xsrf': 'elastalert'
},
'json': {
'locatorId': 'LEGACY_SHORT_URL_LOCATOR',
'params': {
'url': '/app/dev_tools?security_tenant=global#/console'
}
}
},
expected_url='http://kibana.test.org/goto/a1f77a80-6847-11ec-9b91-e5d43d1e9ca2?security_tenant=global'
)
])
def test_7_16_short_kibana_external_url_formatter(
mock_post: mock.MagicMock,
test_case: ShortenUrlTestCase
):
formatter = ShortKibanaExternalUrlFormatter(
base_url=test_case.base_url,
auth=test_case.authorization,
security_tenant=test_case.security_tenant,
new_shortener=True,
)

actualUrl = formatter.format(test_case.relative_url)
Expand All @@ -214,6 +345,7 @@ def test_short_kinbana_external_url_formatter_request_exception(mock_post: mock.
base_url='http://kibana.test.org',
auth=None,
security_tenant=None,
new_shortener=False,
)
with pytest.raises(EAException, match="Failed to invoke Kibana Shorten URL API"):
formatter.format('http://wacky.org')
Expand Down