From fa0cca443671b33310d89c271e883dbb5d6ced7a Mon Sep 17 00:00:00 2001 From: Obdulia Losantos Date: Wed, 16 Sep 2020 16:35:03 +0200 Subject: [PATCH] feat(kong): multiple BASE_HOST for the same app (#66) --- gateway-manager/src/manage_kong.py | 70 ++++++++++++++------ gateway-manager/templates/cors_template.json | 2 +- 2 files changed, 51 insertions(+), 21 deletions(-) diff --git a/gateway-manager/src/manage_kong.py b/gateway-manager/src/manage_kong.py index ecc8ece..2422f61 100644 --- a/gateway-manager/src/manage_kong.py +++ b/gateway-manager/src/manage_kong.py @@ -142,22 +142,18 @@ def _add_service(config): try: if not _check_404(f'{KONG_INTERNAL_URL}/services/{name}'): LOGGER.warning(f'Service "{name}" already exists!') - return - - # Register service - service_data = { - 'name': name, - 'url': host, - } - service_info = request(method='post', url=f'{KONG_INTERNAL_URL}/services/', data=service_data) - service_id = service_info['id'] - LOGGER.success(f'Added service "{name}": {service_id}') + else: + # Register service + service_data = { + 'name': name, + 'url': host, + } + service_info = request(method='post', url=f'{KONG_INTERNAL_URL}/services/', data=service_data) + service_id = service_info['id'] + LOGGER.success(f'Added service "{name}": {service_id}') # Add CORS plugin for whole domain - cors_data = load_json_file(TEMPLATES['cors'], {'host': BASE_HOST}) - PLUGIN_URL = f'{KONG_INTERNAL_URL}/services/{name}/plugins' - request(method='post', url=PLUGIN_URL, data=cors_data) - LOGGER.success(f'Added CORS plugin to service "{name}"') + _add_service_plugin(name) # Add the public routes (non realm dependant) context = { @@ -166,18 +162,21 @@ def _add_service(config): } paths = [fill_template(p, context) for p in config.get('paths', [])] if paths: - # check route - if not _check_404(f'{KONG_INTERNAL_URL}/services/{name}/routes/{name}'): - LOGGER.warning(f'Route "{name}" for service "{name}" already exists!') + # check route (route names contain only [A-Za-z0-9_]) + route_name = f'{BASE_DOMAIN}__{name}' + route_name = ''.join([c if c.isalnum() else '_' for c in route_name]) + if not _check_404(f'{KONG_INTERNAL_URL}/services/{name}/routes/{route_name}'): + LOGGER.warning(f'Route "{route_name}" for service "{name}" already exists!') return route_data = { - 'name': name, + 'name': route_name, 'hosts': [BASE_DOMAIN, ], 'preserve_host': 'true', 'paths': paths, 'strip_path': config.get('strip_path', 'false'), 'regex_priority': config.get('regex_priority', 0), + 'tags': [BASE_DOMAIN, ] # use tags to identify assigned host } ROUTE_URL = f'{KONG_INTERNAL_URL}/services/{name}/routes' @@ -189,6 +188,36 @@ def _add_service(config): raise e +def _add_service_plugin(name): + # Add CORS plugin for whole domain + # get all plugins, if CORS already added, put BASE_HOST in the list of available origins + cors_data = load_json_file(TEMPLATES['cors'], {'host': BASE_HOST}) + PLUGIN_URL = f'{KONG_INTERNAL_URL}/services/{name}/plugins' + + next_url = PLUGIN_URL + while next_url: + res = request(method='get', url=next_url) + next_url = res['next'] + + for plugin in res['data']: + if plugin['name'] == 'cors': + # update origins list + if isinstance(cors_data['config.origins'], str): + cors_data['config.origins'] = [cors_data['config.origins'], ] + cors_data['config.origins'] += plugin['config']['origins'] + # remove possible duplicates + cors_data['config.origins'] = list(set(cors_data['config.origins'])) + + # remove entry + _plugin_id = plugin['id'] + request(method='delete', url=f'{PLUGIN_URL}/{_plugin_id}') + next_url = None + break + + request(method='post', url=PLUGIN_URL, data=cors_data) + LOGGER.success(f'Added CORS plugin to service "{name}"') + + def _remove_service_and_routes(name, routes_fn=None): LOGGER.info(f'Removing service "{name}"...') @@ -255,7 +284,8 @@ def add_service(service_config, realm, oidc_client): for ep in service_config.get(f'{ep_type}_endpoints', []): ep_name = ep['name'] - route_name = f'{name}__{ep_type}__{ep_name}__{realm}' + route_name = f'{BASE_DOMAIN}__{name}__{ep_type}__{ep_name}__{realm}' + route_name = ''.join([c if c.isalnum() else '_' for c in route_name]) # check route if not _check_404(f'{KONG_INTERNAL_URL}/services/{name}/routes/{route_name}'): @@ -270,7 +300,7 @@ def add_service(service_config, realm, oidc_client): 'paths': paths, 'strip_path': ep.get('strip_path', 'false'), 'regex_priority': ep.get('regex_priority', 0), - 'tags': [realm, ] # use tags to identify assigned realm + 'tags': [BASE_DOMAIN, realm, ] # use tags to identify assigned host and realm } try: diff --git a/gateway-manager/templates/cors_template.json b/gateway-manager/templates/cors_template.json index 47a9a3b..850afcb 100644 --- a/gateway-manager/templates/cors_template.json +++ b/gateway-manager/templates/cors_template.json @@ -5,5 +5,5 @@ "config.headers": "Accept, Accept-Version, Content-Disposition, Content-Length, Content-MD5, Content-Type, Date, Authorization", "config.max_age": 3600, "config.methods": ["HEAD", "GET", "POST", "PUT", "PATCH", "DELETE"], - "config.origins": "${host}/*" + "config.origins": ["${host}/*"] }