-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Closes #9: Add data source version extension.
- Loading branch information
Marina Samuel
committed
Sep 14, 2018
1 parent
ffbad5a
commit 51d28c0
Showing
4 changed files
with
192 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,63 @@ | ||
import json | ||
import logging | ||
|
||
from redash_stmo.resources import add_resource | ||
|
||
from redash.models import DataSource | ||
from redash.handlers.base import BaseResource, get_object_or_404 | ||
from redash.permissions import require_access, view_only | ||
|
||
logger = logging.getLogger(__name__) | ||
|
||
|
||
DATASOURCE_VERSION_PARSE_INFO = { | ||
"pg": { | ||
"version_query": "select version()", | ||
"delimiter": " ", | ||
"index": 1 | ||
}, | ||
"redshift": { | ||
"version_query": "select version()", | ||
"delimiter": " ", | ||
"index": -1 | ||
}, | ||
"mysql": { | ||
"version_query": "select version()", | ||
"delimiter": "-", | ||
"index": 0 | ||
} | ||
} | ||
|
||
class DataSourceVersionResource(BaseResource): | ||
def get(self, data_source_id): | ||
data_source = get_object_or_404( | ||
DataSource.get_by_id_and_org, | ||
data_source_id, | ||
self.current_org | ||
) | ||
require_access(data_source.groups, self.current_user, view_only) | ||
version_info = get_data_source_version(data_source.query_runner) | ||
return {"version": version_info} | ||
|
||
def get_data_source_version(query_runner): | ||
parse_info = DATASOURCE_VERSION_PARSE_INFO.get(query_runner.type()) | ||
if parse_info is None: | ||
return None | ||
|
||
data, error = query_runner.run_query(parse_info["version_query"], None) | ||
if error is not None: | ||
logger.error( | ||
"Unable to run version query for %s: %s", query_runner.type(), error) | ||
return None | ||
try: | ||
version = json.loads(data)['rows'][0]['version'] | ||
except (KeyError, IndexError) as err: | ||
logger.exception( | ||
"Unable to parse data source version for %s: %s", query_runner.type(), err) | ||
return None | ||
|
||
version = version.split(parse_info["delimiter"])[parse_info["index"]] | ||
return version | ||
|
||
def datasource_version(app=None): | ||
add_resource(app, DataSourceVersionResource, '/api/data_sources/<data_source_id>/version') |
57 changes: 57 additions & 0 deletions
57
src/redash_stmo/datasource_version/bundle/datasource_version.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
import React from 'react'; | ||
import PropTypes from 'prop-types'; | ||
import { react2angular } from 'react2angular'; | ||
|
||
class DatasourceVersion extends React.Component { | ||
static propTypes = { | ||
clientConfig: PropTypes.object.isRequired, | ||
datasourceId: PropTypes.number.isRequired, | ||
} | ||
|
||
constructor(props) { | ||
super(props); | ||
this.state = { | ||
version: '', | ||
}; | ||
} | ||
|
||
loadURLData() { | ||
fetch(`${this.props.clientConfig.basePath}api/data_sources/${this.props.datasourceId}/version`) | ||
.then((response) => { | ||
if (response.status === 200) { | ||
return response.json(); | ||
} | ||
return {}; | ||
}) | ||
.catch(error => { | ||
console.error(`Error loading data source version: ${error}`); | ||
return {}; | ||
}) | ||
.then((json) => { | ||
this.setState({ version: json.version }); | ||
}); | ||
} | ||
|
||
componentDidMount() { | ||
this.loadURLData(); | ||
} | ||
|
||
componentDidUpdate(prevProps) { | ||
if (this.props.datasourceId !== prevProps.datasourceId) { | ||
this.loadURLData(); | ||
} | ||
} | ||
|
||
render() { | ||
if (!this.state.version) { | ||
return null; | ||
} | ||
return ( | ||
<span>{this.state.version}</span> | ||
); | ||
} | ||
} | ||
|
||
export default function init(ngModule) { | ||
ngModule.component('datasourceVersion', react2angular(DatasourceVersion, ['datasourceId'], ['clientConfig'])); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,70 @@ | ||
import json | ||
import mock | ||
|
||
from tests import BaseTestCase | ||
from flask import Flask | ||
|
||
from redash.models import DataSource | ||
from redash.query_runner.pg import PostgreSQL | ||
from redash_stmo.datasource_version import datasource_version | ||
|
||
|
||
class TestDatasourceVersion(BaseTestCase): | ||
EXPECTED_DOC_URL = "www.example.com" | ||
def setUp(self): | ||
super(TestDatasourceVersion, self).setUp() | ||
self.admin = self.factory.create_admin() | ||
self.data_source = self.factory.create_data_source() | ||
self.patched_run_query = self._setup_mock('redash.query_runner.pg.PostgreSQL.run_query') | ||
self.patched_runner_type = self._setup_mock('redash.query_runner.pg.PostgreSQL.type') | ||
datasource_version(self.app) | ||
|
||
def _setup_mock(self, function_to_patch): | ||
patcher = mock.patch(function_to_patch) | ||
patched_function = patcher.start() | ||
self.addCleanup(patcher.stop) | ||
return patched_function | ||
|
||
def _test_expected_version_returned(self, expected_version, version_string, runner_type): | ||
self.patched_runner_type.return_value = runner_type | ||
self.patched_run_query.return_value = (json.dumps({ | ||
"rows": | ||
[{ "version": version_string.format(version=expected_version) }] | ||
}), None) | ||
rv = self.make_request('get', '/api/data_sources/{}/version'.format(self.data_source.id), user=self.admin) | ||
self.assertEqual(200, rv.status_code) | ||
self.assertEqual(rv.json['version'], expected_version) | ||
|
||
def test_gets_postgres_version(self): | ||
RUNNER_TYPE = "pg" | ||
DATASOURCE_VERSION = "9.5.10" | ||
VERSION_STRING = ( | ||
"PostgreSQL {version} on x86_64-pc-linux-gnu, compiled by gcc " | ||
"(GCC) 4.8.3 20140911 (Red Hat 4.8.3-9), 64-bit" | ||
) | ||
self._test_expected_version_returned(DATASOURCE_VERSION, VERSION_STRING, RUNNER_TYPE) | ||
|
||
def test_gets_redshift_version(self): | ||
RUNNER_TYPE = "redshift" | ||
DATASOURCE_VERSION = "1.0.3688" | ||
VERSION_STRING = ( | ||
"PostgreSQL 8.0.2 on i686-pc-linux-gnu, compiled by GCC " | ||
"gcc (GCC) 3.4.2 20041017 (Red Hat 3.4.2-6.fc3), Redshift {version}" | ||
) | ||
self._test_expected_version_returned(DATASOURCE_VERSION, VERSION_STRING, RUNNER_TYPE) | ||
|
||
def test_gets_mysql_version(self): | ||
RUNNER_TYPE = "mysql" | ||
DATASOURCE_VERSION = "5.7.16" | ||
VERSION_STRING = "{version}-log" | ||
self._test_expected_version_returned(DATASOURCE_VERSION, VERSION_STRING, RUNNER_TYPE) | ||
|
||
def test_unexpected_json(self): | ||
self.patched_runner_type.return_value = "pg" | ||
self.patched_run_query.return_value = (json.dumps({ | ||
"rows": | ||
[{ "bad_json": "foo" }] | ||
}), None) | ||
rv = self.make_request('get', '/api/data_sources/{}/version'.format(self.data_source.id), user=self.admin) | ||
self.assertEqual(200, rv.status_code) | ||
self.assertEqual(rv.json['version'], None) |