Merge pull request #3208 from quay/project/2-spaces
Fix formatting for the config app
This commit is contained in:
commit
8eb7d73f22
28 changed files with 936 additions and 913 deletions
|
@ -2,7 +2,6 @@ import os
|
||||||
import re
|
import re
|
||||||
import subprocess
|
import subprocess
|
||||||
|
|
||||||
|
|
||||||
# Note: this currently points to the directory above, since we're in the quay config_app dir
|
# Note: this currently points to the directory above, since we're in the quay config_app dir
|
||||||
# TODO(config_extract): revert to root directory rather than the one above
|
# TODO(config_extract): revert to root directory rather than the one above
|
||||||
ROOT_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
ROOT_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
||||||
|
@ -15,7 +14,6 @@ TEMPLATE_DIR = os.path.join(ROOT_DIR, 'templates/')
|
||||||
IS_KUBERNETES = 'KUBERNETES_SERVICE_HOST' in os.environ
|
IS_KUBERNETES = 'KUBERNETES_SERVICE_HOST' in os.environ
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def _get_version_number_changelog():
|
def _get_version_number_changelog():
|
||||||
try:
|
try:
|
||||||
with open(os.path.join(ROOT_DIR, 'CHANGELOG.md')) as f:
|
with open(os.path.join(ROOT_DIR, 'CHANGELOG.md')) as f:
|
||||||
|
|
|
@ -27,14 +27,16 @@ config_provider = get_config_provider(OVERRIDE_CONFIG_DIRECTORY, 'config.yaml',
|
||||||
testing=is_testing)
|
testing=is_testing)
|
||||||
|
|
||||||
if is_testing:
|
if is_testing:
|
||||||
from test.testconfig import TestConfig
|
from test.testconfig import TestConfig
|
||||||
logger.debug('Loading test config.')
|
|
||||||
app.config.from_object(TestConfig())
|
logger.debug('Loading test config.')
|
||||||
|
app.config.from_object(TestConfig())
|
||||||
else:
|
else:
|
||||||
from config import DefaultConfig
|
from config import DefaultConfig
|
||||||
logger.debug('Loading default config.')
|
|
||||||
app.config.from_object(DefaultConfig())
|
logger.debug('Loading default config.')
|
||||||
app.teardown_request(database.close_db_filter)
|
app.config.from_object(DefaultConfig())
|
||||||
|
app.teardown_request(database.close_db_filter)
|
||||||
|
|
||||||
# Load the override config via the provider.
|
# Load the override config via the provider.
|
||||||
config_provider.update_app_config(app.config)
|
config_provider.update_app_config(app.config)
|
||||||
|
|
|
@ -3,7 +3,6 @@ from config_app.c_app import app as application
|
||||||
# Bind all of the blueprints
|
# Bind all of the blueprints
|
||||||
import config_web
|
import config_web
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
logging.config.fileConfig(logfile_path(debug=True), disable_existing_loggers=False)
|
logging.config.fileConfig(logfile_path(debug=True), disable_existing_loggers=False)
|
||||||
application.run(port=5000, debug=True, threaded=True, host='0.0.0.0')
|
application.run(port=5000, debug=True, threaded=True, host='0.0.0.0')
|
||||||
|
|
|
@ -31,13 +31,13 @@ api = ApiExceptionHandlingApi()
|
||||||
api.init_app(api_bp)
|
api.init_app(api_bp)
|
||||||
|
|
||||||
def log_action(kind, user_or_orgname, metadata=None, repo=None, repo_name=None):
|
def log_action(kind, user_or_orgname, metadata=None, repo=None, repo_name=None):
|
||||||
if not metadata:
|
if not metadata:
|
||||||
metadata = {}
|
metadata = {}
|
||||||
|
|
||||||
if repo:
|
if repo:
|
||||||
repo_name = repo.name
|
repo_name = repo.name
|
||||||
|
|
||||||
model.log.log_action(kind, user_or_orgname, repo_name, user_or_orgname, request.remote_addr, metadata)
|
model.log.log_action(kind, user_or_orgname, repo_name, user_or_orgname, request.remote_addr, metadata)
|
||||||
|
|
||||||
def format_date(date):
|
def format_date(date):
|
||||||
""" Output an RFC822 date format. """
|
""" Output an RFC822 date format. """
|
||||||
|
|
|
@ -7,247 +7,248 @@ from config_app.c_app import app
|
||||||
from config_app.config_endpoints.api import method_metadata
|
from config_app.config_endpoints.api import method_metadata
|
||||||
from config_app.config_endpoints.common import fully_qualified_name, PARAM_REGEX, TYPE_CONVERTER
|
from config_app.config_endpoints.common import fully_qualified_name, PARAM_REGEX, TYPE_CONVERTER
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
def generate_route_data():
|
def generate_route_data():
|
||||||
include_internal = True
|
include_internal = True
|
||||||
compact = True
|
compact = True
|
||||||
|
|
||||||
def swagger_parameter(name, description, kind='path', param_type='string', required=True,
|
def swagger_parameter(name, description, kind='path', param_type='string', required=True,
|
||||||
enum=None, schema=None):
|
enum=None, schema=None):
|
||||||
# https://github.com/swagger-api/swagger-spec/blob/master/versions/2.0.md#parameterObject
|
# https://github.com/swagger-api/swagger-spec/blob/master/versions/2.0.md#parameterObject
|
||||||
parameter_info = {
|
parameter_info = {
|
||||||
'name': name,
|
'name': name,
|
||||||
'in': kind,
|
'in': kind,
|
||||||
'required': required
|
'required': required
|
||||||
|
}
|
||||||
|
|
||||||
|
if schema:
|
||||||
|
parameter_info['schema'] = {
|
||||||
|
'$ref': '#/definitions/%s' % schema
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
parameter_info['type'] = param_type
|
||||||
|
|
||||||
|
if enum is not None and len(list(enum)) > 0:
|
||||||
|
parameter_info['enum'] = list(enum)
|
||||||
|
|
||||||
|
return parameter_info
|
||||||
|
|
||||||
|
paths = {}
|
||||||
|
models = {}
|
||||||
|
tags = []
|
||||||
|
tags_added = set()
|
||||||
|
operation_ids = set()
|
||||||
|
|
||||||
|
for rule in app.url_map.iter_rules():
|
||||||
|
endpoint_method = app.view_functions[rule.endpoint]
|
||||||
|
|
||||||
|
# Verify that we have a view class for this API method.
|
||||||
|
if not 'view_class' in dir(endpoint_method):
|
||||||
|
continue
|
||||||
|
|
||||||
|
view_class = endpoint_method.view_class
|
||||||
|
|
||||||
|
# Hide the class if it is internal.
|
||||||
|
internal = method_metadata(view_class, 'internal')
|
||||||
|
if not include_internal and internal:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Build the tag.
|
||||||
|
parts = fully_qualified_name(view_class).split('.')
|
||||||
|
tag_name = parts[-2]
|
||||||
|
if not tag_name in tags_added:
|
||||||
|
tags_added.add(tag_name)
|
||||||
|
tags.append({
|
||||||
|
'name': tag_name,
|
||||||
|
'description': (sys.modules[view_class.__module__].__doc__ or '').strip()
|
||||||
|
})
|
||||||
|
|
||||||
|
# Build the Swagger data for the path.
|
||||||
|
swagger_path = PARAM_REGEX.sub(r'{\2}', rule.rule)
|
||||||
|
full_name = fully_qualified_name(view_class)
|
||||||
|
path_swagger = {
|
||||||
|
'x-name': full_name,
|
||||||
|
'x-path': swagger_path,
|
||||||
|
'x-tag': tag_name
|
||||||
|
}
|
||||||
|
|
||||||
|
related_user_res = method_metadata(view_class, 'related_user_resource')
|
||||||
|
if related_user_res is not None:
|
||||||
|
path_swagger['x-user-related'] = fully_qualified_name(related_user_res)
|
||||||
|
|
||||||
|
paths[swagger_path] = path_swagger
|
||||||
|
|
||||||
|
# Add any global path parameters.
|
||||||
|
param_data_map = view_class.__api_path_params if '__api_path_params' in dir(
|
||||||
|
view_class) else {}
|
||||||
|
if param_data_map:
|
||||||
|
path_parameters_swagger = []
|
||||||
|
for path_parameter in param_data_map:
|
||||||
|
description = param_data_map[path_parameter].get('description')
|
||||||
|
path_parameters_swagger.append(swagger_parameter(path_parameter, description))
|
||||||
|
|
||||||
|
path_swagger['parameters'] = path_parameters_swagger
|
||||||
|
|
||||||
|
# Add the individual HTTP operations.
|
||||||
|
method_names = list(rule.methods.difference(['HEAD', 'OPTIONS']))
|
||||||
|
for method_name in method_names:
|
||||||
|
# https://github.com/swagger-api/swagger-spec/blob/master/versions/2.0.md#operation-object
|
||||||
|
method = getattr(view_class, method_name.lower(), None)
|
||||||
|
if method is None:
|
||||||
|
logger.debug('Unable to find method for %s in class %s', method_name, view_class)
|
||||||
|
continue
|
||||||
|
|
||||||
|
operationId = method_metadata(method, 'nickname')
|
||||||
|
operation_swagger = {
|
||||||
|
'operationId': operationId,
|
||||||
|
'parameters': [],
|
||||||
|
}
|
||||||
|
|
||||||
|
if operationId is None:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if operationId in operation_ids:
|
||||||
|
raise Exception('Duplicate operation Id: %s' % operationId)
|
||||||
|
|
||||||
|
operation_ids.add(operationId)
|
||||||
|
|
||||||
|
# Mark the method as internal.
|
||||||
|
internal = method_metadata(method, 'internal')
|
||||||
|
if internal is not None:
|
||||||
|
operation_swagger['x-internal'] = True
|
||||||
|
|
||||||
|
if include_internal:
|
||||||
|
requires_fresh_login = method_metadata(method, 'requires_fresh_login')
|
||||||
|
if requires_fresh_login is not None:
|
||||||
|
operation_swagger['x-requires-fresh-login'] = True
|
||||||
|
|
||||||
|
# Add the path parameters.
|
||||||
|
if rule.arguments:
|
||||||
|
for path_parameter in rule.arguments:
|
||||||
|
description = param_data_map.get(path_parameter, {}).get('description')
|
||||||
|
operation_swagger['parameters'].append(
|
||||||
|
swagger_parameter(path_parameter, description))
|
||||||
|
|
||||||
|
# Add the query parameters.
|
||||||
|
if '__api_query_params' in dir(method):
|
||||||
|
for query_parameter_info in method.__api_query_params:
|
||||||
|
name = query_parameter_info['name']
|
||||||
|
description = query_parameter_info['help']
|
||||||
|
param_type = TYPE_CONVERTER[query_parameter_info['type']]
|
||||||
|
required = query_parameter_info['required']
|
||||||
|
|
||||||
|
operation_swagger['parameters'].append(
|
||||||
|
swagger_parameter(name, description, kind='query',
|
||||||
|
param_type=param_type,
|
||||||
|
required=required,
|
||||||
|
enum=query_parameter_info['choices']))
|
||||||
|
|
||||||
|
# Add the OAuth security block.
|
||||||
|
# https://github.com/swagger-api/swagger-spec/blob/master/versions/2.0.md#securityRequirementObject
|
||||||
|
scope = method_metadata(method, 'oauth2_scope')
|
||||||
|
if scope and not compact:
|
||||||
|
operation_swagger['security'] = [{'oauth2_implicit': [scope.scope]}]
|
||||||
|
|
||||||
|
# Add the responses block.
|
||||||
|
# https://github.com/swagger-api/swagger-spec/blob/master/versions/2.0.md#responsesObject
|
||||||
|
response_schema_name = method_metadata(method, 'response_schema')
|
||||||
|
if not compact:
|
||||||
|
if response_schema_name:
|
||||||
|
models[response_schema_name] = view_class.schemas[response_schema_name]
|
||||||
|
|
||||||
|
models['ApiError'] = {
|
||||||
|
'type': 'object',
|
||||||
|
'properties': {
|
||||||
|
'status': {
|
||||||
|
'type': 'integer',
|
||||||
|
'description': 'Status code of the response.'
|
||||||
|
},
|
||||||
|
'type': {
|
||||||
|
'type': 'string',
|
||||||
|
'description': 'Reference to the type of the error.'
|
||||||
|
},
|
||||||
|
'detail': {
|
||||||
|
'type': 'string',
|
||||||
|
'description': 'Details about the specific instance of the error.'
|
||||||
|
},
|
||||||
|
'title': {
|
||||||
|
'type': 'string',
|
||||||
|
'description': 'Unique error code to identify the type of error.'
|
||||||
|
},
|
||||||
|
'error_message': {
|
||||||
|
'type': 'string',
|
||||||
|
'description': 'Deprecated; alias for detail'
|
||||||
|
},
|
||||||
|
'error_type': {
|
||||||
|
'type': 'string',
|
||||||
|
'description': 'Deprecated; alias for detail'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'required': [
|
||||||
|
'status',
|
||||||
|
'type',
|
||||||
|
'title',
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
if schema:
|
responses = {
|
||||||
parameter_info['schema'] = {
|
'400': {
|
||||||
'$ref': '#/definitions/%s' % schema
|
'description': 'Bad Request',
|
||||||
}
|
},
|
||||||
|
|
||||||
|
'401': {
|
||||||
|
'description': 'Session required',
|
||||||
|
},
|
||||||
|
|
||||||
|
'403': {
|
||||||
|
'description': 'Unauthorized access',
|
||||||
|
},
|
||||||
|
|
||||||
|
'404': {
|
||||||
|
'description': 'Not found',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, body in responses.items():
|
||||||
|
body['schema'] = {'$ref': '#/definitions/ApiError'}
|
||||||
|
|
||||||
|
if method_name == 'DELETE':
|
||||||
|
responses['204'] = {
|
||||||
|
'description': 'Deleted'
|
||||||
|
}
|
||||||
|
elif method_name == 'POST':
|
||||||
|
responses['201'] = {
|
||||||
|
'description': 'Successful creation'
|
||||||
|
}
|
||||||
else:
|
else:
|
||||||
parameter_info['type'] = param_type
|
responses['200'] = {
|
||||||
|
'description': 'Successful invocation'
|
||||||
|
}
|
||||||
|
|
||||||
if enum is not None and len(list(enum)) > 0:
|
if response_schema_name:
|
||||||
parameter_info['enum'] = list(enum)
|
responses['200']['schema'] = {
|
||||||
|
'$ref': '#/definitions/%s' % response_schema_name
|
||||||
return parameter_info
|
|
||||||
|
|
||||||
paths = {}
|
|
||||||
models = {}
|
|
||||||
tags = []
|
|
||||||
tags_added = set()
|
|
||||||
operation_ids = set()
|
|
||||||
|
|
||||||
for rule in app.url_map.iter_rules():
|
|
||||||
endpoint_method = app.view_functions[rule.endpoint]
|
|
||||||
|
|
||||||
# Verify that we have a view class for this API method.
|
|
||||||
if not 'view_class' in dir(endpoint_method):
|
|
||||||
continue
|
|
||||||
|
|
||||||
view_class = endpoint_method.view_class
|
|
||||||
|
|
||||||
# Hide the class if it is internal.
|
|
||||||
internal = method_metadata(view_class, 'internal')
|
|
||||||
if not include_internal and internal:
|
|
||||||
continue
|
|
||||||
|
|
||||||
# Build the tag.
|
|
||||||
parts = fully_qualified_name(view_class).split('.')
|
|
||||||
tag_name = parts[-2]
|
|
||||||
if not tag_name in tags_added:
|
|
||||||
tags_added.add(tag_name)
|
|
||||||
tags.append({
|
|
||||||
'name': tag_name,
|
|
||||||
'description': (sys.modules[view_class.__module__].__doc__ or '').strip()
|
|
||||||
})
|
|
||||||
|
|
||||||
# Build the Swagger data for the path.
|
|
||||||
swagger_path = PARAM_REGEX.sub(r'{\2}', rule.rule)
|
|
||||||
full_name = fully_qualified_name(view_class)
|
|
||||||
path_swagger = {
|
|
||||||
'x-name': full_name,
|
|
||||||
'x-path': swagger_path,
|
|
||||||
'x-tag': tag_name
|
|
||||||
}
|
|
||||||
|
|
||||||
related_user_res = method_metadata(view_class, 'related_user_resource')
|
|
||||||
if related_user_res is not None:
|
|
||||||
path_swagger['x-user-related'] = fully_qualified_name(related_user_res)
|
|
||||||
|
|
||||||
paths[swagger_path] = path_swagger
|
|
||||||
|
|
||||||
# Add any global path parameters.
|
|
||||||
param_data_map = view_class.__api_path_params if '__api_path_params' in dir(view_class) else {}
|
|
||||||
if param_data_map:
|
|
||||||
path_parameters_swagger = []
|
|
||||||
for path_parameter in param_data_map:
|
|
||||||
description = param_data_map[path_parameter].get('description')
|
|
||||||
path_parameters_swagger.append(swagger_parameter(path_parameter, description))
|
|
||||||
|
|
||||||
path_swagger['parameters'] = path_parameters_swagger
|
|
||||||
|
|
||||||
# Add the individual HTTP operations.
|
|
||||||
method_names = list(rule.methods.difference(['HEAD', 'OPTIONS']))
|
|
||||||
for method_name in method_names:
|
|
||||||
# https://github.com/swagger-api/swagger-spec/blob/master/versions/2.0.md#operation-object
|
|
||||||
method = getattr(view_class, method_name.lower(), None)
|
|
||||||
if method is None:
|
|
||||||
logger.debug('Unable to find method for %s in class %s', method_name, view_class)
|
|
||||||
continue
|
|
||||||
|
|
||||||
operationId = method_metadata(method, 'nickname')
|
|
||||||
operation_swagger = {
|
|
||||||
'operationId': operationId,
|
|
||||||
'parameters': [],
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if operationId is None:
|
operation_swagger['responses'] = responses
|
||||||
continue
|
|
||||||
|
|
||||||
if operationId in operation_ids:
|
# Add the request block.
|
||||||
raise Exception('Duplicate operation Id: %s' % operationId)
|
request_schema_name = method_metadata(method, 'request_schema')
|
||||||
|
if request_schema_name and not compact:
|
||||||
|
models[request_schema_name] = view_class.schemas[request_schema_name]
|
||||||
|
|
||||||
operation_ids.add(operationId)
|
operation_swagger['parameters'].append(
|
||||||
|
swagger_parameter('body', 'Request body contents.', kind='body',
|
||||||
|
schema=request_schema_name))
|
||||||
|
|
||||||
# Mark the method as internal.
|
# Add the operation to the parent path.
|
||||||
internal = method_metadata(method, 'internal')
|
if not internal or (internal and include_internal):
|
||||||
if internal is not None:
|
path_swagger[method_name.lower()] = operation_swagger
|
||||||
operation_swagger['x-internal'] = True
|
|
||||||
|
|
||||||
if include_internal:
|
tags.sort(key=lambda t: t['name'])
|
||||||
requires_fresh_login = method_metadata(method, 'requires_fresh_login')
|
paths = OrderedDict(sorted(paths.items(), key=lambda p: p[1]['x-tag']))
|
||||||
if requires_fresh_login is not None:
|
|
||||||
operation_swagger['x-requires-fresh-login'] = True
|
|
||||||
|
|
||||||
# Add the path parameters.
|
if compact:
|
||||||
if rule.arguments:
|
return {'paths': paths}
|
||||||
for path_parameter in rule.arguments:
|
|
||||||
description = param_data_map.get(path_parameter, {}).get('description')
|
|
||||||
operation_swagger['parameters'].append(swagger_parameter(path_parameter, description))
|
|
||||||
|
|
||||||
# Add the query parameters.
|
|
||||||
if '__api_query_params' in dir(method):
|
|
||||||
for query_parameter_info in method.__api_query_params:
|
|
||||||
name = query_parameter_info['name']
|
|
||||||
description = query_parameter_info['help']
|
|
||||||
param_type = TYPE_CONVERTER[query_parameter_info['type']]
|
|
||||||
required = query_parameter_info['required']
|
|
||||||
|
|
||||||
operation_swagger['parameters'].append(
|
|
||||||
swagger_parameter(name, description, kind='query',
|
|
||||||
param_type=param_type,
|
|
||||||
required=required,
|
|
||||||
enum=query_parameter_info['choices']))
|
|
||||||
|
|
||||||
# Add the OAuth security block.
|
|
||||||
# https://github.com/swagger-api/swagger-spec/blob/master/versions/2.0.md#securityRequirementObject
|
|
||||||
scope = method_metadata(method, 'oauth2_scope')
|
|
||||||
if scope and not compact:
|
|
||||||
operation_swagger['security'] = [{'oauth2_implicit': [scope.scope]}]
|
|
||||||
|
|
||||||
# Add the responses block.
|
|
||||||
# https://github.com/swagger-api/swagger-spec/blob/master/versions/2.0.md#responsesObject
|
|
||||||
response_schema_name = method_metadata(method, 'response_schema')
|
|
||||||
if not compact:
|
|
||||||
if response_schema_name:
|
|
||||||
models[response_schema_name] = view_class.schemas[response_schema_name]
|
|
||||||
|
|
||||||
models['ApiError'] = {
|
|
||||||
'type': 'object',
|
|
||||||
'properties': {
|
|
||||||
'status': {
|
|
||||||
'type': 'integer',
|
|
||||||
'description': 'Status code of the response.'
|
|
||||||
},
|
|
||||||
'type': {
|
|
||||||
'type': 'string',
|
|
||||||
'description': 'Reference to the type of the error.'
|
|
||||||
},
|
|
||||||
'detail': {
|
|
||||||
'type': 'string',
|
|
||||||
'description': 'Details about the specific instance of the error.'
|
|
||||||
},
|
|
||||||
'title': {
|
|
||||||
'type': 'string',
|
|
||||||
'description': 'Unique error code to identify the type of error.'
|
|
||||||
},
|
|
||||||
'error_message': {
|
|
||||||
'type': 'string',
|
|
||||||
'description': 'Deprecated; alias for detail'
|
|
||||||
},
|
|
||||||
'error_type': {
|
|
||||||
'type': 'string',
|
|
||||||
'description': 'Deprecated; alias for detail'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
'required': [
|
|
||||||
'status',
|
|
||||||
'type',
|
|
||||||
'title',
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
responses = {
|
|
||||||
'400': {
|
|
||||||
'description': 'Bad Request',
|
|
||||||
},
|
|
||||||
|
|
||||||
'401': {
|
|
||||||
'description': 'Session required',
|
|
||||||
},
|
|
||||||
|
|
||||||
'403': {
|
|
||||||
'description': 'Unauthorized access',
|
|
||||||
},
|
|
||||||
|
|
||||||
'404': {
|
|
||||||
'description': 'Not found',
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, body in responses.items():
|
|
||||||
body['schema'] = {'$ref': '#/definitions/ApiError'}
|
|
||||||
|
|
||||||
if method_name == 'DELETE':
|
|
||||||
responses['204'] = {
|
|
||||||
'description': 'Deleted'
|
|
||||||
}
|
|
||||||
elif method_name == 'POST':
|
|
||||||
responses['201'] = {
|
|
||||||
'description': 'Successful creation'
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
responses['200'] = {
|
|
||||||
'description': 'Successful invocation'
|
|
||||||
}
|
|
||||||
|
|
||||||
if response_schema_name:
|
|
||||||
responses['200']['schema'] = {
|
|
||||||
'$ref': '#/definitions/%s' % response_schema_name
|
|
||||||
}
|
|
||||||
|
|
||||||
operation_swagger['responses'] = responses
|
|
||||||
|
|
||||||
# Add the request block.
|
|
||||||
request_schema_name = method_metadata(method, 'request_schema')
|
|
||||||
if request_schema_name and not compact:
|
|
||||||
models[request_schema_name] = view_class.schemas[request_schema_name]
|
|
||||||
|
|
||||||
operation_swagger['parameters'].append(
|
|
||||||
swagger_parameter('body', 'Request body contents.', kind='body',
|
|
||||||
schema=request_schema_name))
|
|
||||||
|
|
||||||
# Add the operation to the parent path.
|
|
||||||
if not internal or (internal and include_internal):
|
|
||||||
path_swagger[method_name.lower()] = operation_swagger
|
|
||||||
|
|
||||||
tags.sort(key=lambda t: t['name'])
|
|
||||||
paths = OrderedDict(sorted(paths.items(), key=lambda p: p[1]['x-tag']))
|
|
||||||
|
|
||||||
if compact:
|
|
||||||
return {'paths': paths}
|
|
||||||
|
|
|
@ -3,7 +3,8 @@ import logging
|
||||||
from flask import abort, request
|
from flask import abort, request
|
||||||
|
|
||||||
from config_app.config_endpoints.api.suconfig_models_pre_oci import pre_oci_model as model
|
from config_app.config_endpoints.api.suconfig_models_pre_oci import pre_oci_model as model
|
||||||
from config_app.config_endpoints.api import resource, ApiResource, nickname, validate_json_request, kubernetes_only
|
from config_app.config_endpoints.api import resource, ApiResource, nickname, validate_json_request, \
|
||||||
|
kubernetes_only
|
||||||
from config_app.c_app import (app, config_provider, superusers, ip_resolver,
|
from config_app.c_app import (app, config_provider, superusers, ip_resolver,
|
||||||
instance_keys, INIT_SCRIPTS_LOCATION)
|
instance_keys, INIT_SCRIPTS_LOCATION)
|
||||||
from config_app.config_util.k8saccessor import KubernetesAccessorSingleton
|
from config_app.config_util.k8saccessor import KubernetesAccessorSingleton
|
||||||
|
@ -11,7 +12,8 @@ from config_app.config_util.k8saccessor import KubernetesAccessorSingleton
|
||||||
from data.database import configure
|
from data.database import configure
|
||||||
from data.runmigration import run_alembic_migration
|
from data.runmigration import run_alembic_migration
|
||||||
from util.config.configutil import add_enterprise_config_defaults
|
from util.config.configutil import add_enterprise_config_defaults
|
||||||
from util.config.validator import validate_service_for_config, ValidatorContext, is_valid_config_upload_filename
|
from util.config.validator import validate_service_for_config, ValidatorContext, \
|
||||||
|
is_valid_config_upload_filename
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -85,6 +87,7 @@ class SuperUserRegistryStatus(ApiResource):
|
||||||
""" Resource for determining the status of the registry, such as if config exists,
|
""" Resource for determining the status of the registry, such as if config exists,
|
||||||
if a database is configured, and if it has any defined users.
|
if a database is configured, and if it has any defined users.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@nickname('scRegistryStatus')
|
@nickname('scRegistryStatus')
|
||||||
def get(self):
|
def get(self):
|
||||||
""" Returns the status of the registry. """
|
""" Returns the status of the registry. """
|
||||||
|
@ -121,6 +124,7 @@ class _AlembicLogHandler(logging.Handler):
|
||||||
@resource('/v1/superuser/setupdb')
|
@resource('/v1/superuser/setupdb')
|
||||||
class SuperUserSetupDatabase(ApiResource):
|
class SuperUserSetupDatabase(ApiResource):
|
||||||
""" Resource for invoking alembic to setup the database. """
|
""" Resource for invoking alembic to setup the database. """
|
||||||
|
|
||||||
@nickname('scSetupDatabase')
|
@nickname('scSetupDatabase')
|
||||||
def get(self):
|
def get(self):
|
||||||
""" Invokes the alembic upgrade process. """
|
""" Invokes the alembic upgrade process. """
|
||||||
|
@ -251,7 +255,8 @@ class SuperUserConfigValidate(ApiResource):
|
||||||
# so we also allow it to be called if there is no valid registry configuration setup. Note that
|
# so we also allow it to be called if there is no valid registry configuration setup. Note that
|
||||||
# this is also safe since this method does not access any information not given in the request.
|
# this is also safe since this method does not access any information not given in the request.
|
||||||
config = request.get_json()['config']
|
config = request.get_json()['config']
|
||||||
validator_context = ValidatorContext.from_app(app, config, request.get_json().get('password', ''),
|
validator_context = ValidatorContext.from_app(app, config,
|
||||||
|
request.get_json().get('password', ''),
|
||||||
instance_keys=instance_keys,
|
instance_keys=instance_keys,
|
||||||
ip_resolver=ip_resolver,
|
ip_resolver=ip_resolver,
|
||||||
config_provider=config_provider,
|
config_provider=config_provider,
|
||||||
|
@ -294,6 +299,7 @@ class SuperUserKubernetesDeployment(ApiResource):
|
||||||
@resource('/v1/superuser/config/kubernetes')
|
@resource('/v1/superuser/config/kubernetes')
|
||||||
class SuperUserKubernetesConfiguration(ApiResource):
|
class SuperUserKubernetesConfiguration(ApiResource):
|
||||||
""" Resource for saving the config files to kubernetes secrets. """
|
""" Resource for saving the config files to kubernetes secrets. """
|
||||||
|
|
||||||
@kubernetes_only
|
@kubernetes_only
|
||||||
@nickname('scDeployConfiguration')
|
@nickname('scDeployConfiguration')
|
||||||
def post(self):
|
def post(self):
|
||||||
|
@ -303,6 +309,7 @@ class SuperUserKubernetesConfiguration(ApiResource):
|
||||||
@resource('/v1/superuser/config/file/<filename>')
|
@resource('/v1/superuser/config/file/<filename>')
|
||||||
class SuperUserConfigFile(ApiResource):
|
class SuperUserConfigFile(ApiResource):
|
||||||
""" Resource for fetching the status of config files and overriding them. """
|
""" Resource for fetching the status of config files and overriding them. """
|
||||||
|
|
||||||
@nickname('scConfigFileExists')
|
@nickname('scConfigFileExists')
|
||||||
def get(self, filename):
|
def get(self, filename):
|
||||||
""" Returns whether the configuration file with the given name exists. """
|
""" Returns whether the configuration file with the given name exists. """
|
||||||
|
@ -313,7 +320,6 @@ class SuperUserConfigFile(ApiResource):
|
||||||
'exists': config_provider.volume_file_exists(filename)
|
'exists': config_provider.volume_file_exists(filename)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@nickname('scUpdateConfigFile')
|
@nickname('scUpdateConfigFile')
|
||||||
def post(self, filename):
|
def post(self, filename):
|
||||||
""" Updates the configuration file with the given name. """
|
""" Updates the configuration file with the given name. """
|
||||||
|
|
|
@ -4,36 +4,36 @@ from six import add_metaclass
|
||||||
|
|
||||||
@add_metaclass(ABCMeta)
|
@add_metaclass(ABCMeta)
|
||||||
class SuperuserConfigDataInterface(object):
|
class SuperuserConfigDataInterface(object):
|
||||||
|
"""
|
||||||
|
Interface that represents all data store interactions required by the superuser config API.
|
||||||
|
"""
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def is_valid(self):
|
||||||
"""
|
"""
|
||||||
Interface that represents all data store interactions required by the superuser config API.
|
Returns true if the configured database is valid.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def is_valid(self):
|
def has_users(self):
|
||||||
"""
|
"""
|
||||||
Returns true if the configured database is valid.
|
Returns true if there are any users defined.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def has_users(self):
|
def create_superuser(self, username, password, email):
|
||||||
"""
|
"""
|
||||||
Returns true if there are any users defined.
|
Creates a new superuser with the given username, password and email. Returns the user's UUID.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def create_superuser(self, username, password, email):
|
def has_federated_login(self, username, service_name):
|
||||||
"""
|
"""
|
||||||
Creates a new superuser with the given username, password and email. Returns the user's UUID.
|
Returns true if the matching user has a federated login under the matching service.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def has_federated_login(self, username, service_name):
|
def attach_federated_login(self, username, service_name, federated_username):
|
||||||
"""
|
"""
|
||||||
Returns true if the matching user has a federated login under the matching service.
|
Attaches a federatated login to the matching user, under the given service.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def attach_federated_login(self, username, service_name, federated_username):
|
|
||||||
"""
|
|
||||||
Attaches a federatated login to the matching user, under the given service.
|
|
||||||
"""
|
|
||||||
|
|
|
@ -4,34 +4,34 @@ from config_app.config_endpoints.api.suconfig_models_interface import SuperuserC
|
||||||
|
|
||||||
|
|
||||||
class PreOCIModel(SuperuserConfigDataInterface):
|
class PreOCIModel(SuperuserConfigDataInterface):
|
||||||
# Note: this method is different than has_users: the user select will throw if the user
|
# Note: this method is different than has_users: the user select will throw if the user
|
||||||
# table does not exist, whereas has_users assumes the table is valid
|
# table does not exist, whereas has_users assumes the table is valid
|
||||||
def is_valid(self):
|
def is_valid(self):
|
||||||
try:
|
try:
|
||||||
list(User.select().limit(1))
|
list(User.select().limit(1))
|
||||||
return True
|
return True
|
||||||
except:
|
except:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def has_users(self):
|
def has_users(self):
|
||||||
return bool(list(User.select().limit(1)))
|
return bool(list(User.select().limit(1)))
|
||||||
|
|
||||||
def create_superuser(self, username, password, email):
|
def create_superuser(self, username, password, email):
|
||||||
return model.user.create_user(username, password, email, auto_verify=True).uuid
|
return model.user.create_user(username, password, email, auto_verify=True).uuid
|
||||||
|
|
||||||
def has_federated_login(self, username, service_name):
|
def has_federated_login(self, username, service_name):
|
||||||
user = model.user.get_user(username)
|
user = model.user.get_user(username)
|
||||||
if user is None:
|
if user is None:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
return bool(model.user.lookup_federated_login(user, service_name))
|
return bool(model.user.lookup_federated_login(user, service_name))
|
||||||
|
|
||||||
def attach_federated_login(self, username, service_name, federated_username):
|
def attach_federated_login(self, username, service_name, federated_username):
|
||||||
user = model.user.get_user(username)
|
user = model.user.get_user(username)
|
||||||
if user is None:
|
if user is None:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
model.user.attach_federated_login(user, service_name, federated_username)
|
model.user.attach_federated_login(user, service_name, federated_username)
|
||||||
|
|
||||||
|
|
||||||
pre_oci_model = PreOCIModel()
|
pre_oci_model = PreOCIModel()
|
||||||
|
|
|
@ -6,165 +6,168 @@ from config_app.config_endpoints.api import format_date
|
||||||
|
|
||||||
|
|
||||||
def user_view(user):
|
def user_view(user):
|
||||||
return {
|
return {
|
||||||
'name': user.username,
|
'name': user.username,
|
||||||
'kind': 'user',
|
'kind': 'user',
|
||||||
'is_robot': user.robot,
|
'is_robot': user.robot,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class RepositoryBuild(namedtuple('RepositoryBuild',
|
class RepositoryBuild(namedtuple('RepositoryBuild',
|
||||||
['uuid', 'logs_archived', 'repository_namespace_user_username', 'repository_name',
|
['uuid', 'logs_archived', 'repository_namespace_user_username',
|
||||||
'can_write', 'can_read', 'pull_robot', 'resource_key', 'trigger', 'display_name',
|
'repository_name',
|
||||||
'started', 'job_config', 'phase', 'status', 'error', 'archive_url'])):
|
'can_write', 'can_read', 'pull_robot', 'resource_key', 'trigger',
|
||||||
"""
|
'display_name',
|
||||||
RepositoryBuild represents a build associated with a repostiory
|
'started', 'job_config', 'phase', 'status', 'error',
|
||||||
:type uuid: string
|
'archive_url'])):
|
||||||
:type logs_archived: boolean
|
"""
|
||||||
:type repository_namespace_user_username: string
|
RepositoryBuild represents a build associated with a repostiory
|
||||||
:type repository_name: string
|
:type uuid: string
|
||||||
:type can_write: boolean
|
:type logs_archived: boolean
|
||||||
:type can_write: boolean
|
:type repository_namespace_user_username: string
|
||||||
:type pull_robot: User
|
:type repository_name: string
|
||||||
:type resource_key: string
|
:type can_write: boolean
|
||||||
:type trigger: Trigger
|
:type can_write: boolean
|
||||||
:type display_name: string
|
:type pull_robot: User
|
||||||
:type started: boolean
|
:type resource_key: string
|
||||||
:type job_config: {Any -> Any}
|
:type trigger: Trigger
|
||||||
:type phase: string
|
:type display_name: string
|
||||||
:type status: string
|
:type started: boolean
|
||||||
:type error: string
|
:type job_config: {Any -> Any}
|
||||||
:type archive_url: string
|
:type phase: string
|
||||||
"""
|
:type status: string
|
||||||
|
:type error: string
|
||||||
|
:type archive_url: string
|
||||||
|
"""
|
||||||
|
|
||||||
def to_dict(self):
|
def to_dict(self):
|
||||||
|
|
||||||
resp = {
|
resp = {
|
||||||
'id': self.uuid,
|
'id': self.uuid,
|
||||||
'phase': self.phase,
|
'phase': self.phase,
|
||||||
'started': format_date(self.started),
|
'started': format_date(self.started),
|
||||||
'display_name': self.display_name,
|
'display_name': self.display_name,
|
||||||
'status': self.status or {},
|
'status': self.status or {},
|
||||||
'subdirectory': self.job_config.get('build_subdir', ''),
|
'subdirectory': self.job_config.get('build_subdir', ''),
|
||||||
'dockerfile_path': self.job_config.get('build_subdir', ''),
|
'dockerfile_path': self.job_config.get('build_subdir', ''),
|
||||||
'context': self.job_config.get('context', ''),
|
'context': self.job_config.get('context', ''),
|
||||||
'tags': self.job_config.get('docker_tags', []),
|
'tags': self.job_config.get('docker_tags', []),
|
||||||
'manual_user': self.job_config.get('manual_user', None),
|
'manual_user': self.job_config.get('manual_user', None),
|
||||||
'is_writer': self.can_write,
|
'is_writer': self.can_write,
|
||||||
'trigger': self.trigger.to_dict(),
|
'trigger': self.trigger.to_dict(),
|
||||||
'trigger_metadata': self.job_config.get('trigger_metadata', None) if self.can_read else None,
|
'trigger_metadata': self.job_config.get('trigger_metadata', None) if self.can_read else None,
|
||||||
'resource_key': self.resource_key,
|
'resource_key': self.resource_key,
|
||||||
'pull_robot': user_view(self.pull_robot) if self.pull_robot else None,
|
'pull_robot': user_view(self.pull_robot) if self.pull_robot else None,
|
||||||
'repository': {
|
'repository': {
|
||||||
'namespace': self.repository_namespace_user_username,
|
'namespace': self.repository_namespace_user_username,
|
||||||
'name': self.repository_name
|
'name': self.repository_name
|
||||||
},
|
},
|
||||||
'error': self.error,
|
'error': self.error,
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.can_write:
|
if self.can_write:
|
||||||
if self.resource_key is not None:
|
if self.resource_key is not None:
|
||||||
resp['archive_url'] = self.archive_url
|
resp['archive_url'] = self.archive_url
|
||||||
elif self.job_config.get('archive_url', None):
|
elif self.job_config.get('archive_url', None):
|
||||||
resp['archive_url'] = self.job_config['archive_url']
|
resp['archive_url'] = self.job_config['archive_url']
|
||||||
|
|
||||||
return resp
|
return resp
|
||||||
|
|
||||||
|
|
||||||
class Approval(namedtuple('Approval', ['approver', 'approval_type', 'approved_date', 'notes'])):
|
class Approval(namedtuple('Approval', ['approver', 'approval_type', 'approved_date', 'notes'])):
|
||||||
"""
|
"""
|
||||||
Approval represents whether a key has been approved or not
|
Approval represents whether a key has been approved or not
|
||||||
:type approver: User
|
:type approver: User
|
||||||
:type approval_type: string
|
:type approval_type: string
|
||||||
:type approved_date: Date
|
:type approved_date: Date
|
||||||
:type notes: string
|
:type notes: string
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def to_dict(self):
|
def to_dict(self):
|
||||||
return {
|
return {
|
||||||
'approver': self.approver.to_dict() if self.approver else None,
|
'approver': self.approver.to_dict() if self.approver else None,
|
||||||
'approval_type': self.approval_type,
|
'approval_type': self.approval_type,
|
||||||
'approved_date': self.approved_date,
|
'approved_date': self.approved_date,
|
||||||
'notes': self.notes,
|
'notes': self.notes,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class ServiceKey(namedtuple('ServiceKey', ['name', 'kid', 'service', 'jwk', 'metadata', 'created_date',
|
class ServiceKey(
|
||||||
'expiration_date', 'rotation_duration', 'approval'])):
|
namedtuple('ServiceKey', ['name', 'kid', 'service', 'jwk', 'metadata', 'created_date',
|
||||||
"""
|
'expiration_date', 'rotation_duration', 'approval'])):
|
||||||
ServiceKey is an apostille signing key
|
"""
|
||||||
:type name: string
|
ServiceKey is an apostille signing key
|
||||||
:type kid: int
|
:type name: string
|
||||||
:type service: string
|
:type kid: int
|
||||||
:type jwk: string
|
:type service: string
|
||||||
:type metadata: string
|
:type jwk: string
|
||||||
:type created_date: Date
|
:type metadata: string
|
||||||
:type expiration_date: Date
|
:type created_date: Date
|
||||||
:type rotation_duration: Date
|
:type expiration_date: Date
|
||||||
:type approval: Approval
|
:type rotation_duration: Date
|
||||||
|
:type approval: Approval
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def to_dict(self):
|
def to_dict(self):
|
||||||
return {
|
return {
|
||||||
'name': self.name,
|
'name': self.name,
|
||||||
'kid': self.kid,
|
'kid': self.kid,
|
||||||
'service': self.service,
|
'service': self.service,
|
||||||
'jwk': self.jwk,
|
'jwk': self.jwk,
|
||||||
'metadata': self.metadata,
|
'metadata': self.metadata,
|
||||||
'created_date': self.created_date,
|
'created_date': self.created_date,
|
||||||
'expiration_date': self.expiration_date,
|
'expiration_date': self.expiration_date,
|
||||||
'rotation_duration': self.rotation_duration,
|
'rotation_duration': self.rotation_duration,
|
||||||
'approval': self.approval.to_dict() if self.approval is not None else None,
|
'approval': self.approval.to_dict() if self.approval is not None else None,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class User(namedtuple('User', ['username', 'email', 'verified', 'enabled', 'robot'])):
|
class User(namedtuple('User', ['username', 'email', 'verified', 'enabled', 'robot'])):
|
||||||
"""
|
"""
|
||||||
User represents a single user.
|
User represents a single user.
|
||||||
:type username: string
|
:type username: string
|
||||||
:type email: string
|
:type email: string
|
||||||
:type verified: boolean
|
:type verified: boolean
|
||||||
:type enabled: boolean
|
:type enabled: boolean
|
||||||
:type robot: User
|
:type robot: User
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def to_dict(self):
|
def to_dict(self):
|
||||||
user_data = {
|
user_data = {
|
||||||
'kind': 'user',
|
'kind': 'user',
|
||||||
'name': self.username,
|
'name': self.username,
|
||||||
'username': self.username,
|
'username': self.username,
|
||||||
'email': self.email,
|
'email': self.email,
|
||||||
'verified': self.verified,
|
'verified': self.verified,
|
||||||
'enabled': self.enabled,
|
'enabled': self.enabled,
|
||||||
}
|
}
|
||||||
|
|
||||||
return user_data
|
return user_data
|
||||||
|
|
||||||
|
|
||||||
class Organization(namedtuple('Organization', ['username', 'email'])):
|
class Organization(namedtuple('Organization', ['username', 'email'])):
|
||||||
"""
|
"""
|
||||||
Organization represents a single org.
|
Organization represents a single org.
|
||||||
:type username: string
|
:type username: string
|
||||||
:type email: string
|
:type email: string
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def to_dict(self):
|
|
||||||
return {
|
|
||||||
'name': self.username,
|
|
||||||
'email': self.email,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
return {
|
||||||
|
'name': self.username,
|
||||||
|
'email': self.email,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@add_metaclass(ABCMeta)
|
@add_metaclass(ABCMeta)
|
||||||
class SuperuserDataInterface(object):
|
class SuperuserDataInterface(object):
|
||||||
|
"""
|
||||||
|
Interface that represents all data store interactions required by a superuser api.
|
||||||
|
"""
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def list_all_service_keys(self):
|
||||||
"""
|
"""
|
||||||
Interface that represents all data store interactions required by a superuser api.
|
Returns a list of service keys
|
||||||
"""
|
"""
|
||||||
@abstractmethod
|
|
||||||
def list_all_service_keys(self):
|
|
||||||
"""
|
|
||||||
Returns a list of service keys
|
|
||||||
"""
|
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
from data import model
|
from data import model
|
||||||
|
|
||||||
from config_app.config_endpoints.api.superuser_models_interface import SuperuserDataInterface, User, ServiceKey, Approval
|
from config_app.config_endpoints.api.superuser_models_interface import (SuperuserDataInterface, User, ServiceKey,
|
||||||
|
Approval)
|
||||||
|
|
||||||
|
|
||||||
def _create_user(user):
|
def _create_user(user):
|
||||||
if user is None:
|
if user is None:
|
||||||
|
@ -11,12 +13,15 @@ def _create_user(user):
|
||||||
def _create_key(key):
|
def _create_key(key):
|
||||||
approval = None
|
approval = None
|
||||||
if key.approval is not None:
|
if key.approval is not None:
|
||||||
approval = Approval(_create_user(key.approval.approver), key.approval.approval_type, key.approval.approved_date,
|
approval = Approval(_create_user(key.approval.approver), key.approval.approval_type,
|
||||||
|
key.approval.approved_date,
|
||||||
key.approval.notes)
|
key.approval.notes)
|
||||||
|
|
||||||
return ServiceKey(key.name, key.kid, key.service, key.jwk, key.metadata, key.created_date, key.expiration_date,
|
return ServiceKey(key.name, key.kid, key.service, key.jwk, key.metadata, key.created_date,
|
||||||
|
key.expiration_date,
|
||||||
key.rotation_duration, approval)
|
key.rotation_duration, approval)
|
||||||
|
|
||||||
|
|
||||||
class ServiceKeyDoesNotExist(Exception):
|
class ServiceKeyDoesNotExist(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -30,6 +35,7 @@ class PreOCIModel(SuperuserDataInterface):
|
||||||
PreOCIModel implements the data model for the SuperUser using a database schema
|
PreOCIModel implements the data model for the SuperUser using a database schema
|
||||||
before it was changed to support the OCI specification.
|
before it was changed to support the OCI specification.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def list_all_service_keys(self):
|
def list_all_service_keys(self):
|
||||||
keys = model.service_keys.list_all_keys()
|
keys = model.service_keys.list_all_keys()
|
||||||
return [_create_key(key) for key in keys]
|
return [_create_key(key) for key in keys]
|
||||||
|
@ -43,8 +49,10 @@ class PreOCIModel(SuperuserDataInterface):
|
||||||
except model.ServiceKeyAlreadyApproved:
|
except model.ServiceKeyAlreadyApproved:
|
||||||
raise ServiceKeyAlreadyApproved
|
raise ServiceKeyAlreadyApproved
|
||||||
|
|
||||||
def generate_service_key(self, service, expiration_date, kid=None, name='', metadata=None, rotation_duration=None):
|
def generate_service_key(self, service, expiration_date, kid=None, name='', metadata=None,
|
||||||
(private_key, key) = model.service_keys.generate_service_key(service, expiration_date, metadata=metadata, name=name)
|
rotation_duration=None):
|
||||||
|
(private_key, key) = model.service_keys.generate_service_key(service, expiration_date,
|
||||||
|
metadata=metadata, name=name)
|
||||||
|
|
||||||
return private_key, key.kid
|
return private_key, key.kid
|
||||||
|
|
||||||
|
|
|
@ -10,50 +10,51 @@ from config_app.c_app import app, config_provider
|
||||||
from config_app.config_endpoints.api import resource, ApiResource, nickname
|
from config_app.config_endpoints.api import resource, ApiResource, nickname
|
||||||
from config_app.config_util.tar import tarinfo_filter_partial, strip_absolute_path_and_add_trailing_dir
|
from config_app.config_util.tar import tarinfo_filter_partial, strip_absolute_path_and_add_trailing_dir
|
||||||
|
|
||||||
|
|
||||||
@resource('/v1/configapp/initialization')
|
@resource('/v1/configapp/initialization')
|
||||||
class ConfigInitialization(ApiResource):
|
class ConfigInitialization(ApiResource):
|
||||||
"""
|
"""
|
||||||
Resource for dealing with any initialization logic for the config app
|
Resource for dealing with any initialization logic for the config app
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@nickname('scStartNewConfig')
|
@nickname('scStartNewConfig')
|
||||||
def post(self):
|
def post(self):
|
||||||
config_provider.new_config_dir()
|
config_provider.new_config_dir()
|
||||||
return make_response('OK')
|
return make_response('OK')
|
||||||
|
|
||||||
|
|
||||||
@resource('/v1/configapp/tarconfig')
|
@resource('/v1/configapp/tarconfig')
|
||||||
class TarConfigLoader(ApiResource):
|
class TarConfigLoader(ApiResource):
|
||||||
"""
|
"""
|
||||||
Resource for dealing with configuration as a tarball,
|
Resource for dealing with configuration as a tarball,
|
||||||
including loading and generating functions
|
including loading and generating functions
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@nickname('scGetConfigTarball')
|
@nickname('scGetConfigTarball')
|
||||||
def get(self):
|
def get(self):
|
||||||
config_path = config_provider.get_config_dir_path()
|
config_path = config_provider.get_config_dir_path()
|
||||||
tar_dir_prefix = strip_absolute_path_and_add_trailing_dir(config_path)
|
tar_dir_prefix = strip_absolute_path_and_add_trailing_dir(config_path)
|
||||||
temp = tempfile.NamedTemporaryFile()
|
temp = tempfile.NamedTemporaryFile()
|
||||||
|
|
||||||
tar = tarfile.open(temp.name, mode="w|gz")
|
tar = tarfile.open(temp.name, mode="w|gz")
|
||||||
for name in os.listdir(config_path):
|
for name in os.listdir(config_path):
|
||||||
tar.add(os.path.join(config_path, name), filter=tarinfo_filter_partial(tar_dir_prefix))
|
tar.add(os.path.join(config_path, name), filter=tarinfo_filter_partial(tar_dir_prefix))
|
||||||
|
|
||||||
tar.close()
|
tar.close()
|
||||||
return send_file(temp.name, mimetype='application/gzip')
|
return send_file(temp.name, mimetype='application/gzip')
|
||||||
|
|
||||||
@nickname('scUploadTarballConfig')
|
@nickname('scUploadTarballConfig')
|
||||||
def put(self):
|
def put(self):
|
||||||
""" Loads tarball config into the config provider """
|
""" Loads tarball config into the config provider """
|
||||||
# Generate a new empty dir to load the config into
|
# Generate a new empty dir to load the config into
|
||||||
config_provider.new_config_dir()
|
config_provider.new_config_dir()
|
||||||
input_stream = request.stream
|
input_stream = request.stream
|
||||||
with tarfile.open(mode="r|gz", fileobj=input_stream) as tar_stream:
|
with tarfile.open(mode="r|gz", fileobj=input_stream) as tar_stream:
|
||||||
tar_stream.extractall(config_provider.get_config_dir_path())
|
tar_stream.extractall(config_provider.get_config_dir_path())
|
||||||
|
|
||||||
# now try to connect to the db provided in their config to validate it works
|
# now try to connect to the db provided in their config to validate it works
|
||||||
combined = dict(**app.config)
|
combined = dict(**app.config)
|
||||||
combined.update(config_provider.get_config())
|
combined.update(config_provider.get_config())
|
||||||
configure(combined)
|
configure(combined)
|
||||||
|
|
||||||
return make_response('OK')
|
return make_response('OK')
|
||||||
|
|
|
@ -5,15 +5,14 @@ from config_app.config_endpoints.api.superuser_models_interface import user_view
|
||||||
|
|
||||||
@resource('/v1/user/')
|
@resource('/v1/user/')
|
||||||
class User(ApiResource):
|
class User(ApiResource):
|
||||||
""" Operations related to users. """
|
""" Operations related to users. """
|
||||||
|
|
||||||
@nickname('getLoggedInUser')
|
@nickname('getLoggedInUser')
|
||||||
def get(self):
|
def get(self):
|
||||||
""" Get user information for the authenticated user. """
|
""" Get user information for the authenticated user. """
|
||||||
user = get_authenticated_user()
|
user = get_authenticated_user()
|
||||||
# TODO(config): figure out if we need user validation
|
# TODO(config): figure out if we need user validation
|
||||||
# if user is None or user.organization or not UserReadPermission(user.username).can():
|
# if user is None or user.organization or not UserReadPermission(user.username).can():
|
||||||
# raise InvalidToken("Requires authentication", payload={'session_required': False})
|
# raise InvalidToken("Requires authentication", payload={'session_required': False})
|
||||||
|
|
||||||
return user_view(user)
|
|
||||||
|
|
||||||
|
return user_view(user)
|
||||||
|
|
|
@ -13,52 +13,50 @@ from config_app.config_util.k8sconfig import get_k8s_namespace
|
||||||
|
|
||||||
|
|
||||||
def truthy_bool(param):
|
def truthy_bool(param):
|
||||||
return param not in {False, 'false', 'False', '0', 'FALSE', '', 'null'}
|
return param not in {False, 'false', 'False', '0', 'FALSE', '', 'null'}
|
||||||
|
|
||||||
|
|
||||||
DEFAULT_JS_BUNDLE_NAME = 'configapp'
|
DEFAULT_JS_BUNDLE_NAME = 'configapp'
|
||||||
PARAM_REGEX = re.compile(r'<([^:>]+:)*([\w]+)>')
|
PARAM_REGEX = re.compile(r'<([^:>]+:)*([\w]+)>')
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
TYPE_CONVERTER = {
|
TYPE_CONVERTER = {
|
||||||
truthy_bool: 'boolean',
|
truthy_bool: 'boolean',
|
||||||
str: 'string',
|
str: 'string',
|
||||||
basestring: 'string',
|
basestring: 'string',
|
||||||
reqparse.text_type: 'string',
|
reqparse.text_type: 'string',
|
||||||
int: 'integer',
|
int: 'integer',
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
def _list_files(path, extension, contains=""):
|
def _list_files(path, extension, contains=""):
|
||||||
""" Returns a list of all the files with the given extension found under the given path. """
|
""" Returns a list of all the files with the given extension found under the given path. """
|
||||||
|
|
||||||
def matches(f):
|
def matches(f):
|
||||||
return os.path.splitext(f)[1] == '.' + extension and contains in os.path.splitext(f)[0]
|
return os.path.splitext(f)[1] == '.' + extension and contains in os.path.splitext(f)[0]
|
||||||
|
|
||||||
def join_path(dp, f):
|
def join_path(dp, f):
|
||||||
# Remove the static/ prefix. It is added in the template.
|
# Remove the static/ prefix. It is added in the template.
|
||||||
return os.path.join(dp, f)[len(ROOT_DIR) + 1 + len('config_app/static/'):]
|
return os.path.join(dp, f)[len(ROOT_DIR) + 1 + len('config_app/static/'):]
|
||||||
|
|
||||||
filepath = os.path.join(os.path.join(ROOT_DIR, 'config_app/static/'), path)
|
filepath = os.path.join(os.path.join(ROOT_DIR, 'config_app/static/'), path)
|
||||||
return [join_path(dp, f) for dp, _, files in os.walk(filepath) for f in files if matches(f)]
|
return [join_path(dp, f) for dp, _, files in os.walk(filepath) for f in files if matches(f)]
|
||||||
|
|
||||||
|
|
||||||
def render_page_template(name, route_data=None, js_bundle_name=DEFAULT_JS_BUNDLE_NAME, **kwargs):
|
def render_page_template(name, route_data=None, js_bundle_name=DEFAULT_JS_BUNDLE_NAME, **kwargs):
|
||||||
""" Renders the page template with the given name as the response and returns its contents. """
|
""" Renders the page template with the given name as the response and returns its contents. """
|
||||||
main_scripts = _list_files('build', 'js', js_bundle_name)
|
main_scripts = _list_files('build', 'js', js_bundle_name)
|
||||||
|
|
||||||
contents = render_template(name,
|
contents = render_template(name,
|
||||||
route_data=route_data,
|
route_data=route_data,
|
||||||
main_scripts=main_scripts,
|
main_scripts=main_scripts,
|
||||||
config_set=frontend_visible_config(app.config),
|
config_set=frontend_visible_config(app.config),
|
||||||
kubernetes_namespace=IS_KUBERNETES and get_k8s_namespace(),
|
kubernetes_namespace=IS_KUBERNETES and get_k8s_namespace(),
|
||||||
**kwargs)
|
**kwargs)
|
||||||
|
|
||||||
resp = make_response(contents)
|
resp = make_response(contents)
|
||||||
resp.headers['X-FRAME-OPTIONS'] = 'DENY'
|
resp.headers['X-FRAME-OPTIONS'] = 'DENY'
|
||||||
return resp
|
return resp
|
||||||
|
|
||||||
|
|
||||||
def fully_qualified_name(method_view_class):
|
def fully_qualified_name(method_view_class):
|
||||||
return '%s.%s' % (method_view_class.__module__, method_view_class.__name__)
|
return '%s.%s' % (method_view_class.__module__, method_view_class.__name__)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -5,63 +5,62 @@ from werkzeug.exceptions import HTTPException
|
||||||
|
|
||||||
|
|
||||||
class ApiErrorType(Enum):
|
class ApiErrorType(Enum):
|
||||||
invalid_request = 'invalid_request'
|
invalid_request = 'invalid_request'
|
||||||
|
|
||||||
|
|
||||||
class ApiException(HTTPException):
|
class ApiException(HTTPException):
|
||||||
"""
|
"""
|
||||||
Represents an error in the application/problem+json format.
|
Represents an error in the application/problem+json format.
|
||||||
|
|
||||||
See: https://tools.ietf.org/html/rfc7807
|
See: https://tools.ietf.org/html/rfc7807
|
||||||
|
|
||||||
- "type" (string) - A URI reference that identifies the
|
- "type" (string) - A URI reference that identifies the
|
||||||
problem type.
|
problem type.
|
||||||
|
|
||||||
- "title" (string) - A short, human-readable summary of the problem
|
- "title" (string) - A short, human-readable summary of the problem
|
||||||
type. It SHOULD NOT change from occurrence to occurrence of the
|
type. It SHOULD NOT change from occurrence to occurrence of the
|
||||||
problem, except for purposes of localization
|
problem, except for purposes of localization
|
||||||
|
|
||||||
- "status" (number) - The HTTP status code
|
- "status" (number) - The HTTP status code
|
||||||
|
|
||||||
- "detail" (string) - A human-readable explanation specific to this
|
- "detail" (string) - A human-readable explanation specific to this
|
||||||
occurrence of the problem.
|
occurrence of the problem.
|
||||||
|
|
||||||
- "instance" (string) - A URI reference that identifies the specific
|
- "instance" (string) - A URI reference that identifies the specific
|
||||||
occurrence of the problem. It may or may not yield further
|
occurrence of the problem. It may or may not yield further
|
||||||
information if dereferenced.
|
information if dereferenced.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, error_type, status_code, error_description, payload=None):
|
def __init__(self, error_type, status_code, error_description, payload=None):
|
||||||
Exception.__init__(self)
|
Exception.__init__(self)
|
||||||
self.error_description = error_description
|
self.error_description = error_description
|
||||||
self.code = status_code
|
self.code = status_code
|
||||||
self.payload = payload
|
self.payload = payload
|
||||||
self.error_type = error_type
|
self.error_type = error_type
|
||||||
self.data = self.to_dict()
|
self.data = self.to_dict()
|
||||||
|
|
||||||
super(ApiException, self).__init__(error_description, None)
|
super(ApiException, self).__init__(error_description, None)
|
||||||
|
|
||||||
def to_dict(self):
|
def to_dict(self):
|
||||||
rv = dict(self.payload or ())
|
rv = dict(self.payload or ())
|
||||||
|
|
||||||
if self.error_description is not None:
|
if self.error_description is not None:
|
||||||
rv['detail'] = self.error_description
|
rv['detail'] = self.error_description
|
||||||
rv['error_message'] = self.error_description # TODO: deprecate
|
rv['error_message'] = self.error_description # TODO: deprecate
|
||||||
|
|
||||||
rv['error_type'] = self.error_type.value # TODO: deprecate
|
rv['error_type'] = self.error_type.value # TODO: deprecate
|
||||||
rv['title'] = self.error_type.value
|
rv['title'] = self.error_type.value
|
||||||
rv['type'] = url_for('api.error', error_type=self.error_type.value, _external=True)
|
rv['type'] = url_for('api.error', error_type=self.error_type.value, _external=True)
|
||||||
rv['status'] = self.code
|
rv['status'] = self.code
|
||||||
|
|
||||||
return rv
|
|
||||||
|
|
||||||
|
return rv
|
||||||
|
|
||||||
|
|
||||||
class InvalidRequest(ApiException):
|
class InvalidRequest(ApiException):
|
||||||
def __init__(self, error_description, payload=None):
|
def __init__(self, error_description, payload=None):
|
||||||
ApiException.__init__(self, ApiErrorType.invalid_request, 400, error_description, payload)
|
ApiException.__init__(self, ApiErrorType.invalid_request, 400, error_description, payload)
|
||||||
|
|
||||||
|
|
||||||
class InvalidResponse(ApiException):
|
class InvalidResponse(ApiException):
|
||||||
def __init__(self, error_description, payload=None):
|
def __init__(self, error_description, payload=None):
|
||||||
ApiException.__init__(self, ApiErrorType.invalid_response, 400, error_description, payload)
|
ApiException.__init__(self, ApiErrorType.invalid_response, 400, error_description, payload)
|
||||||
|
|
|
@ -5,22 +5,19 @@ from config_app.config_endpoints.common import render_page_template
|
||||||
from config_app.config_endpoints.api.discovery import generate_route_data
|
from config_app.config_endpoints.api.discovery import generate_route_data
|
||||||
from config_app.config_endpoints.api import no_cache
|
from config_app.config_endpoints.api import no_cache
|
||||||
|
|
||||||
|
|
||||||
setup_web = Blueprint('setup_web', __name__, template_folder='templates')
|
setup_web = Blueprint('setup_web', __name__, template_folder='templates')
|
||||||
|
|
||||||
|
|
||||||
@lru_cache(maxsize=1)
|
@lru_cache(maxsize=1)
|
||||||
def _get_route_data():
|
def _get_route_data():
|
||||||
return generate_route_data()
|
return generate_route_data()
|
||||||
|
|
||||||
|
|
||||||
def render_page_template_with_routedata(name, *args, **kwargs):
|
def render_page_template_with_routedata(name, *args, **kwargs):
|
||||||
return render_page_template(name, _get_route_data(), *args, **kwargs)
|
return render_page_template(name, _get_route_data(), *args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
@no_cache
|
@no_cache
|
||||||
@setup_web.route('/', methods=['GET'], defaults={'path': ''})
|
@setup_web.route('/', methods=['GET'], defaults={'path': ''})
|
||||||
def index(path, **kwargs):
|
def index(path, **kwargs):
|
||||||
return render_page_template_with_routedata('index.html', js_bundle_name='configapp', **kwargs)
|
return render_page_template_with_routedata('index.html', js_bundle_name='configapp', **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -6,41 +6,42 @@ from config_app.config_util.k8saccessor import KubernetesAccessorSingleton
|
||||||
|
|
||||||
|
|
||||||
class TransientDirectoryProvider(FileConfigProvider):
|
class TransientDirectoryProvider(FileConfigProvider):
|
||||||
""" Implementation of the config provider that reads and writes the data
|
""" Implementation of the config provider that reads and writes the data
|
||||||
from/to the file system, only using temporary directories,
|
from/to the file system, only using temporary directories,
|
||||||
deleting old dirs and creating new ones as requested.
|
deleting old dirs and creating new ones as requested.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, config_volume, yaml_filename, py_filename):
|
||||||
|
# Create a temp directory that will be cleaned up when we change the config path
|
||||||
|
# This should ensure we have no "pollution" of different configs:
|
||||||
|
# no uploaded config should ever affect subsequent config modifications/creations
|
||||||
|
temp_dir = TemporaryDirectory()
|
||||||
|
self.temp_dir = temp_dir
|
||||||
|
super(TransientDirectoryProvider, self).__init__(temp_dir.name, yaml_filename, py_filename)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def provider_id(self):
|
||||||
|
return 'transient'
|
||||||
|
|
||||||
|
def new_config_dir(self):
|
||||||
"""
|
"""
|
||||||
def __init__(self, config_volume, yaml_filename, py_filename):
|
Update the path with a new temporary directory, deleting the old one in the process
|
||||||
# Create a temp directory that will be cleaned up when we change the config path
|
"""
|
||||||
# This should ensure we have no "pollution" of different configs:
|
self.temp_dir.cleanup()
|
||||||
# no uploaded config should ever affect subsequent config modifications/creations
|
temp_dir = TemporaryDirectory()
|
||||||
temp_dir = TemporaryDirectory()
|
|
||||||
self.temp_dir = temp_dir
|
|
||||||
super(TransientDirectoryProvider, self).__init__(temp_dir.name, yaml_filename, py_filename)
|
|
||||||
|
|
||||||
@property
|
self.config_volume = temp_dir.name
|
||||||
def provider_id(self):
|
self.temp_dir = temp_dir
|
||||||
return 'transient'
|
self.yaml_path = os.path.join(temp_dir.name, self.yaml_filename)
|
||||||
|
|
||||||
def new_config_dir(self):
|
def get_config_dir_path(self):
|
||||||
"""
|
return self.config_volume
|
||||||
Update the path with a new temporary directory, deleting the old one in the process
|
|
||||||
"""
|
|
||||||
self.temp_dir.cleanup()
|
|
||||||
temp_dir = TemporaryDirectory()
|
|
||||||
|
|
||||||
self.config_volume = temp_dir.name
|
def save_configuration_to_kubernetes(self):
|
||||||
self.temp_dir = temp_dir
|
config_path = self.get_config_dir_path()
|
||||||
self.yaml_path = os.path.join(temp_dir.name, self.yaml_filename)
|
|
||||||
|
|
||||||
def get_config_dir_path(self):
|
for name in os.listdir(config_path):
|
||||||
return self.config_volume
|
file_path = os.path.join(self.config_volume, name)
|
||||||
|
KubernetesAccessorSingleton.get_instance().save_file_as_secret(name, file_path)
|
||||||
|
|
||||||
def save_configuration_to_kubernetes(self):
|
return 200
|
||||||
config_path = self.get_config_dir_path()
|
|
||||||
|
|
||||||
for name in os.listdir(config_path):
|
|
||||||
file_path = os.path.join(self.config_volume, name)
|
|
||||||
KubernetesAccessorSingleton.get_instance().save_file_as_secret(name, file_path)
|
|
||||||
|
|
||||||
return 200
|
|
||||||
|
|
|
@ -4,9 +4,9 @@ from config_app.config_util.config.TransientDirectoryProvider import TransientDi
|
||||||
|
|
||||||
|
|
||||||
def get_config_provider(config_volume, yaml_filename, py_filename, testing=False):
|
def get_config_provider(config_volume, yaml_filename, py_filename, testing=False):
|
||||||
""" Loads and returns the config provider for the current environment. """
|
""" Loads and returns the config provider for the current environment. """
|
||||||
|
|
||||||
if testing:
|
if testing:
|
||||||
return TestConfigProvider()
|
return TestConfigProvider()
|
||||||
|
|
||||||
return TransientDirectoryProvider(config_volume, yaml_filename, py_filename)
|
return TransientDirectoryProvider(config_volume, yaml_filename, py_filename)
|
||||||
|
|
|
@ -8,64 +8,65 @@ logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class BaseFileProvider(BaseProvider):
|
class BaseFileProvider(BaseProvider):
|
||||||
""" Base implementation of the config provider that reads the data from the file system. """
|
""" Base implementation of the config provider that reads the data from the file system. """
|
||||||
def __init__(self, config_volume, yaml_filename, py_filename):
|
|
||||||
self.config_volume = config_volume
|
|
||||||
self.yaml_filename = yaml_filename
|
|
||||||
self.py_filename = py_filename
|
|
||||||
|
|
||||||
self.yaml_path = os.path.join(config_volume, yaml_filename)
|
def __init__(self, config_volume, yaml_filename, py_filename):
|
||||||
self.py_path = os.path.join(config_volume, py_filename)
|
self.config_volume = config_volume
|
||||||
|
self.yaml_filename = yaml_filename
|
||||||
|
self.py_filename = py_filename
|
||||||
|
|
||||||
def update_app_config(self, app_config):
|
self.yaml_path = os.path.join(config_volume, yaml_filename)
|
||||||
if os.path.exists(self.py_path):
|
self.py_path = os.path.join(config_volume, py_filename)
|
||||||
logger.debug('Applying config file: %s', self.py_path)
|
|
||||||
app_config.from_pyfile(self.py_path)
|
|
||||||
|
|
||||||
if os.path.exists(self.yaml_path):
|
def update_app_config(self, app_config):
|
||||||
logger.debug('Applying config file: %s', self.yaml_path)
|
if os.path.exists(self.py_path):
|
||||||
import_yaml(app_config, self.yaml_path)
|
logger.debug('Applying config file: %s', self.py_path)
|
||||||
|
app_config.from_pyfile(self.py_path)
|
||||||
|
|
||||||
def get_config(self):
|
if os.path.exists(self.yaml_path):
|
||||||
if not self.config_exists():
|
logger.debug('Applying config file: %s', self.yaml_path)
|
||||||
return None
|
import_yaml(app_config, self.yaml_path)
|
||||||
|
|
||||||
config_obj = {}
|
def get_config(self):
|
||||||
import_yaml(config_obj, self.yaml_path)
|
if not self.config_exists():
|
||||||
return config_obj
|
return None
|
||||||
|
|
||||||
def config_exists(self):
|
config_obj = {}
|
||||||
return self.volume_file_exists(self.yaml_filename)
|
import_yaml(config_obj, self.yaml_path)
|
||||||
|
return config_obj
|
||||||
|
|
||||||
def volume_exists(self):
|
def config_exists(self):
|
||||||
return os.path.exists(self.config_volume)
|
return self.volume_file_exists(self.yaml_filename)
|
||||||
|
|
||||||
def volume_file_exists(self, filename):
|
def volume_exists(self):
|
||||||
return os.path.exists(os.path.join(self.config_volume, filename))
|
return os.path.exists(self.config_volume)
|
||||||
|
|
||||||
def get_volume_file(self, filename, mode='r'):
|
def volume_file_exists(self, filename):
|
||||||
return open(os.path.join(self.config_volume, filename), mode=mode)
|
return os.path.exists(os.path.join(self.config_volume, filename))
|
||||||
|
|
||||||
def get_volume_path(self, directory, filename):
|
def get_volume_file(self, filename, mode='r'):
|
||||||
return os.path.join(directory, filename)
|
return open(os.path.join(self.config_volume, filename), mode=mode)
|
||||||
|
|
||||||
def list_volume_directory(self, path):
|
def get_volume_path(self, directory, filename):
|
||||||
dirpath = os.path.join(self.config_volume, path)
|
return os.path.join(directory, filename)
|
||||||
if not os.path.exists(dirpath):
|
|
||||||
return None
|
|
||||||
|
|
||||||
if not os.path.isdir(dirpath):
|
def list_volume_directory(self, path):
|
||||||
return None
|
dirpath = os.path.join(self.config_volume, path)
|
||||||
|
if not os.path.exists(dirpath):
|
||||||
|
return None
|
||||||
|
|
||||||
return os.listdir(dirpath)
|
if not os.path.isdir(dirpath):
|
||||||
|
return None
|
||||||
|
|
||||||
def requires_restart(self, app_config):
|
return os.listdir(dirpath)
|
||||||
file_config = self.get_config()
|
|
||||||
if not file_config:
|
|
||||||
return False
|
|
||||||
|
|
||||||
for key in file_config:
|
def requires_restart(self, app_config):
|
||||||
if app_config.get(key) != file_config[key]:
|
file_config = self.get_config()
|
||||||
return True
|
if not file_config:
|
||||||
|
return False
|
||||||
|
|
||||||
return False
|
for key in file_config:
|
||||||
|
if app_config.get(key) != file_config[key]:
|
||||||
|
return True
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
|
@ -4,57 +4,57 @@ import logging
|
||||||
from config_app.config_util.config.baseprovider import export_yaml, CannotWriteConfigException
|
from config_app.config_util.config.baseprovider import export_yaml, CannotWriteConfigException
|
||||||
from config_app.config_util.config.basefileprovider import BaseFileProvider
|
from config_app.config_util.config.basefileprovider import BaseFileProvider
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
def _ensure_parent_dir(filepath):
|
def _ensure_parent_dir(filepath):
|
||||||
""" Ensures that the parent directory of the given file path exists. """
|
""" Ensures that the parent directory of the given file path exists. """
|
||||||
try:
|
try:
|
||||||
parentpath = os.path.abspath(os.path.join(filepath, os.pardir))
|
parentpath = os.path.abspath(os.path.join(filepath, os.pardir))
|
||||||
if not os.path.isdir(parentpath):
|
if not os.path.isdir(parentpath):
|
||||||
os.makedirs(parentpath)
|
os.makedirs(parentpath)
|
||||||
except IOError as ioe:
|
except IOError as ioe:
|
||||||
raise CannotWriteConfigException(str(ioe))
|
raise CannotWriteConfigException(str(ioe))
|
||||||
|
|
||||||
|
|
||||||
class FileConfigProvider(BaseFileProvider):
|
class FileConfigProvider(BaseFileProvider):
|
||||||
""" Implementation of the config provider that reads and writes the data
|
""" Implementation of the config provider that reads and writes the data
|
||||||
from/to the file system. """
|
from/to the file system. """
|
||||||
def __init__(self, config_volume, yaml_filename, py_filename):
|
|
||||||
super(FileConfigProvider, self).__init__(config_volume, yaml_filename, py_filename)
|
|
||||||
|
|
||||||
@property
|
def __init__(self, config_volume, yaml_filename, py_filename):
|
||||||
def provider_id(self):
|
super(FileConfigProvider, self).__init__(config_volume, yaml_filename, py_filename)
|
||||||
return 'file'
|
|
||||||
|
|
||||||
def save_config(self, config_obj):
|
@property
|
||||||
export_yaml(config_obj, self.yaml_path)
|
def provider_id(self):
|
||||||
|
return 'file'
|
||||||
|
|
||||||
def write_volume_file(self, filename, contents):
|
def save_config(self, config_obj):
|
||||||
filepath = os.path.join(self.config_volume, filename)
|
export_yaml(config_obj, self.yaml_path)
|
||||||
_ensure_parent_dir(filepath)
|
|
||||||
|
|
||||||
try:
|
def write_volume_file(self, filename, contents):
|
||||||
with open(filepath, mode='w') as f:
|
filepath = os.path.join(self.config_volume, filename)
|
||||||
f.write(contents)
|
_ensure_parent_dir(filepath)
|
||||||
except IOError as ioe:
|
|
||||||
raise CannotWriteConfigException(str(ioe))
|
|
||||||
|
|
||||||
return filepath
|
try:
|
||||||
|
with open(filepath, mode='w') as f:
|
||||||
|
f.write(contents)
|
||||||
|
except IOError as ioe:
|
||||||
|
raise CannotWriteConfigException(str(ioe))
|
||||||
|
|
||||||
def remove_volume_file(self, filename):
|
return filepath
|
||||||
filepath = os.path.join(self.config_volume, filename)
|
|
||||||
os.remove(filepath)
|
|
||||||
|
|
||||||
def save_volume_file(self, filename, flask_file):
|
def remove_volume_file(self, filename):
|
||||||
filepath = os.path.join(self.config_volume, filename)
|
filepath = os.path.join(self.config_volume, filename)
|
||||||
_ensure_parent_dir(filepath)
|
os.remove(filepath)
|
||||||
|
|
||||||
# Write the file.
|
def save_volume_file(self, filename, flask_file):
|
||||||
try:
|
filepath = os.path.join(self.config_volume, filename)
|
||||||
flask_file.save(filepath)
|
_ensure_parent_dir(filepath)
|
||||||
except IOError as ioe:
|
|
||||||
raise CannotWriteConfigException(str(ioe))
|
|
||||||
|
|
||||||
return filepath
|
# Write the file.
|
||||||
|
try:
|
||||||
|
flask_file.save(filepath)
|
||||||
|
except IOError as ioe:
|
||||||
|
raise CannotWriteConfigException(str(ioe))
|
||||||
|
|
||||||
|
return filepath
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
import json
|
import json
|
||||||
import io
|
import io
|
||||||
import os
|
import os
|
||||||
from datetime import datetime, timedelta
|
|
||||||
|
|
||||||
from config_app.config_util.config.baseprovider import BaseProvider
|
from config_app.config_util.config.baseprovider import BaseProvider
|
||||||
|
|
||||||
|
@ -9,73 +8,73 @@ REAL_FILES = ['test/data/signing-private.gpg', 'test/data/signing-public.gpg', '
|
||||||
|
|
||||||
|
|
||||||
class TestConfigProvider(BaseProvider):
|
class TestConfigProvider(BaseProvider):
|
||||||
""" Implementation of the config provider for testing. Everything is kept in-memory instead on
|
""" Implementation of the config provider for testing. Everything is kept in-memory instead on
|
||||||
the real file system. """
|
the real file system. """
|
||||||
def __init__(self):
|
|
||||||
self.clear()
|
|
||||||
|
|
||||||
def clear(self):
|
def __init__(self):
|
||||||
self.files = {}
|
self.clear()
|
||||||
self._config = {}
|
|
||||||
|
|
||||||
@property
|
def clear(self):
|
||||||
def provider_id(self):
|
self.files = {}
|
||||||
return 'test'
|
self._config = {}
|
||||||
|
|
||||||
def update_app_config(self, app_config):
|
@property
|
||||||
self._config = app_config
|
def provider_id(self):
|
||||||
|
return 'test'
|
||||||
|
|
||||||
def get_config(self):
|
def update_app_config(self, app_config):
|
||||||
if not 'config.yaml' in self.files:
|
self._config = app_config
|
||||||
return None
|
|
||||||
|
|
||||||
return json.loads(self.files.get('config.yaml', '{}'))
|
def get_config(self):
|
||||||
|
if not 'config.yaml' in self.files:
|
||||||
|
return None
|
||||||
|
|
||||||
def save_config(self, config_obj):
|
return json.loads(self.files.get('config.yaml', '{}'))
|
||||||
self.files['config.yaml'] = json.dumps(config_obj)
|
|
||||||
|
|
||||||
def config_exists(self):
|
def save_config(self, config_obj):
|
||||||
return 'config.yaml' in self.files
|
self.files['config.yaml'] = json.dumps(config_obj)
|
||||||
|
|
||||||
def volume_exists(self):
|
def config_exists(self):
|
||||||
return True
|
return 'config.yaml' in self.files
|
||||||
|
|
||||||
def volume_file_exists(self, filename):
|
def volume_exists(self):
|
||||||
if filename in REAL_FILES:
|
return True
|
||||||
return True
|
|
||||||
|
|
||||||
return filename in self.files
|
def volume_file_exists(self, filename):
|
||||||
|
if filename in REAL_FILES:
|
||||||
|
return True
|
||||||
|
|
||||||
def save_volume_file(self, filename, flask_file):
|
return filename in self.files
|
||||||
self.files[filename] = flask_file.read()
|
|
||||||
|
|
||||||
def write_volume_file(self, filename, contents):
|
def save_volume_file(self, filename, flask_file):
|
||||||
self.files[filename] = contents
|
self.files[filename] = flask_file.read()
|
||||||
|
|
||||||
def get_volume_file(self, filename, mode='r'):
|
def write_volume_file(self, filename, contents):
|
||||||
if filename in REAL_FILES:
|
self.files[filename] = contents
|
||||||
return open(filename, mode=mode)
|
|
||||||
|
|
||||||
return io.BytesIO(self.files[filename])
|
def get_volume_file(self, filename, mode='r'):
|
||||||
|
if filename in REAL_FILES:
|
||||||
|
return open(filename, mode=mode)
|
||||||
|
|
||||||
def remove_volume_file(self, filename):
|
return io.BytesIO(self.files[filename])
|
||||||
self.files.pop(filename, None)
|
|
||||||
|
|
||||||
def list_volume_directory(self, path):
|
def remove_volume_file(self, filename):
|
||||||
paths = []
|
self.files.pop(filename, None)
|
||||||
for filename in self.files:
|
|
||||||
if filename.startswith(path):
|
|
||||||
paths.append(filename[len(path)+1:])
|
|
||||||
|
|
||||||
return paths
|
def list_volume_directory(self, path):
|
||||||
|
paths = []
|
||||||
|
for filename in self.files:
|
||||||
|
if filename.startswith(path):
|
||||||
|
paths.append(filename[len(path) + 1:])
|
||||||
|
|
||||||
def requires_restart(self, app_config):
|
return paths
|
||||||
return False
|
|
||||||
|
|
||||||
def reset_for_test(self):
|
def requires_restart(self, app_config):
|
||||||
self._config['SUPER_USERS'] = ['devtable']
|
return False
|
||||||
self.files = {}
|
|
||||||
|
|
||||||
def get_volume_path(self, directory, filename):
|
def reset_for_test(self):
|
||||||
return os.path.join(directory, filename)
|
self._config['SUPER_USERS'] = ['devtable']
|
||||||
|
self.files = {}
|
||||||
|
|
||||||
|
def get_volume_path(self, directory, filename):
|
||||||
|
return os.path.join(directory, filename)
|
||||||
|
|
|
@ -11,135 +11,135 @@ logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
QE_DEPLOYMENT_LABEL = 'quay-enterprise-component'
|
QE_DEPLOYMENT_LABEL = 'quay-enterprise-component'
|
||||||
|
|
||||||
|
|
||||||
class KubernetesAccessorSingleton(object):
|
class KubernetesAccessorSingleton(object):
|
||||||
""" Singleton allowing access to kubernetes operations """
|
""" Singleton allowing access to kubernetes operations """
|
||||||
_instance = None
|
_instance = None
|
||||||
|
|
||||||
def __init__(self, kube_config=None):
|
def __init__(self, kube_config=None):
|
||||||
self.kube_config = kube_config
|
self.kube_config = kube_config
|
||||||
if kube_config is None:
|
if kube_config is None:
|
||||||
self.kube_config = KubernetesConfig.from_env()
|
self.kube_config = KubernetesConfig.from_env()
|
||||||
|
|
||||||
KubernetesAccessorSingleton._instance = self
|
KubernetesAccessorSingleton._instance = self
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def get_instance(cls, kube_config=None):
|
def get_instance(cls, kube_config=None):
|
||||||
"""
|
"""
|
||||||
Singleton getter implementation, returns the instance if one exists, otherwise creates the
|
Singleton getter implementation, returns the instance if one exists, otherwise creates the
|
||||||
instance and ties it to the class.
|
instance and ties it to the class.
|
||||||
:return: KubernetesAccessorSingleton
|
:return: KubernetesAccessorSingleton
|
||||||
"""
|
"""
|
||||||
if cls._instance is None:
|
if cls._instance is None:
|
||||||
return cls(kube_config)
|
return cls(kube_config)
|
||||||
|
|
||||||
return cls._instance
|
return cls._instance
|
||||||
|
|
||||||
def save_file_as_secret(self, name, file_path):
|
def save_file_as_secret(self, name, file_path):
|
||||||
with open(file_path) as f:
|
with open(file_path) as f:
|
||||||
value = f.read()
|
value = f.read()
|
||||||
self._update_secret_file(name, value)
|
self._update_secret_file(name, value)
|
||||||
|
|
||||||
def get_qe_deployments(self):
|
def get_qe_deployments(self):
|
||||||
""""
|
""""
|
||||||
Returns all deployments matching the label selector provided in the KubeConfig
|
Returns all deployments matching the label selector provided in the KubeConfig
|
||||||
"""
|
"""
|
||||||
deployment_selector_url = 'namespaces/%s/deployments?labelSelector=%s%%3D%s' % (
|
deployment_selector_url = 'namespaces/%s/deployments?labelSelector=%s%%3D%s' % (
|
||||||
self.kube_config.qe_namespace, QE_DEPLOYMENT_LABEL, self.kube_config.qe_deployment_selector
|
self.kube_config.qe_namespace, QE_DEPLOYMENT_LABEL, self.kube_config.qe_deployment_selector
|
||||||
)
|
)
|
||||||
|
|
||||||
response = self._execute_k8s_api('GET', deployment_selector_url, api_prefix='apis/extensions/v1beta1')
|
response = self._execute_k8s_api('GET', deployment_selector_url, api_prefix='apis/extensions/v1beta1')
|
||||||
if response.status_code != 200:
|
if response.status_code != 200:
|
||||||
return None
|
return None
|
||||||
return json.loads(response.text)
|
return json.loads(response.text)
|
||||||
|
|
||||||
def cycle_qe_deployments(self, deployment_names):
|
def cycle_qe_deployments(self, deployment_names):
|
||||||
""""
|
""""
|
||||||
Triggers a rollout of all desired deployments in the qe namespace
|
Triggers a rollout of all desired deployments in the qe namespace
|
||||||
"""
|
"""
|
||||||
|
|
||||||
for name in deployment_names:
|
for name in deployment_names:
|
||||||
logger.debug('Cycling deployment %s', name)
|
logger.debug('Cycling deployment %s', name)
|
||||||
deployment_url = 'namespaces/%s/deployments/%s' % (self.kube_config.qe_namespace, name)
|
deployment_url = 'namespaces/%s/deployments/%s' % (self.kube_config.qe_namespace, name)
|
||||||
|
|
||||||
# There is currently no command to simply rolling restart all the pods: https://github.com/kubernetes/kubernetes/issues/13488
|
# There is currently no command to simply rolling restart all the pods: https://github.com/kubernetes/kubernetes/issues/13488
|
||||||
# Instead, we modify the template of the deployment with a dummy env variable to trigger a cycle of the pods
|
# Instead, we modify the template of the deployment with a dummy env variable to trigger a cycle of the pods
|
||||||
# (based off this comment: https://github.com/kubernetes/kubernetes/issues/13488#issuecomment-240393845)
|
# (based off this comment: https://github.com/kubernetes/kubernetes/issues/13488#issuecomment-240393845)
|
||||||
self._assert_success(self._execute_k8s_api('PATCH', deployment_url, {
|
self._assert_success(self._execute_k8s_api('PATCH', deployment_url, {
|
||||||
'spec': {
|
'spec': {
|
||||||
'template': {
|
'template': {
|
||||||
'spec': {
|
'spec': {
|
||||||
'containers': [{
|
'containers': [{
|
||||||
'name': 'quay-enterprise-app', 'env': [{
|
'name': 'quay-enterprise-app', 'env': [{
|
||||||
'name': 'RESTART_TIME',
|
'name': 'RESTART_TIME',
|
||||||
'value': str(datetime.datetime.now())
|
'value': str(datetime.datetime.now())
|
||||||
}]
|
|
||||||
}]
|
}]
|
||||||
}
|
}]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, api_prefix='apis/extensions/v1beta1', content_type='application/strategic-merge-patch+json'))
|
|
||||||
|
|
||||||
|
|
||||||
def _assert_success(self, response):
|
|
||||||
if response.status_code != 200:
|
|
||||||
logger.error('Kubernetes API call failed with response: %s => %s', response.status_code,
|
|
||||||
response.text)
|
|
||||||
raise Exception('Kubernetes API call failed: %s' % response.text)
|
|
||||||
|
|
||||||
def _update_secret_file(self, relative_file_path, value=None):
|
|
||||||
if '/' in relative_file_path:
|
|
||||||
raise Exception('Expected path from get_volume_path, but found slashes')
|
|
||||||
|
|
||||||
# Check first that the namespace for Quay Enterprise exists. If it does not, report that
|
|
||||||
# as an error, as it seems to be a common issue.
|
|
||||||
namespace_url = 'namespaces/%s' % (self.kube_config.qe_namespace)
|
|
||||||
response = self._execute_k8s_api('GET', namespace_url)
|
|
||||||
if response.status_code // 100 != 2:
|
|
||||||
msg = 'A Kubernetes namespace with name `%s` must be created to save config' % self.kube_config.qe_namespace
|
|
||||||
raise Exception(msg)
|
|
||||||
|
|
||||||
# Check if the secret exists. If not, then we create an empty secret and then update the file
|
|
||||||
# inside.
|
|
||||||
secret_url = 'namespaces/%s/secrets/%s' % (self.kube_config.qe_namespace, self.kube_config.qe_config_secret)
|
|
||||||
secret = self._lookup_secret()
|
|
||||||
if secret is None:
|
|
||||||
self._assert_success(self._execute_k8s_api('POST', secret_url, {
|
|
||||||
"kind": "Secret",
|
|
||||||
"apiVersion": "v1",
|
|
||||||
"metadata": {
|
|
||||||
"name": self.kube_config.qe_config_secret
|
|
||||||
},
|
|
||||||
"data": {}
|
|
||||||
}))
|
|
||||||
|
|
||||||
# Update the secret to reflect the file change.
|
|
||||||
secret['data'] = secret.get('data', {})
|
|
||||||
|
|
||||||
if value is not None:
|
|
||||||
secret['data'][relative_file_path] = base64.b64encode(value)
|
|
||||||
else:
|
|
||||||
secret['data'].pop(relative_file_path)
|
|
||||||
|
|
||||||
self._assert_success(self._execute_k8s_api('PUT', secret_url, secret))
|
|
||||||
|
|
||||||
def _lookup_secret(self):
|
|
||||||
secret_url = 'namespaces/%s/secrets/%s' % (self.kube_config.qe_namespace, self.kube_config.qe_config_secret)
|
|
||||||
response = self._execute_k8s_api('GET', secret_url)
|
|
||||||
if response.status_code != 200:
|
|
||||||
return None
|
|
||||||
return json.loads(response.text)
|
|
||||||
|
|
||||||
def _execute_k8s_api(self, method, relative_url, data=None, api_prefix='api/v1', content_type='application/json'):
|
|
||||||
headers = {
|
|
||||||
'Authorization': 'Bearer ' + self.kube_config.service_account_token
|
|
||||||
}
|
}
|
||||||
|
}, api_prefix='apis/extensions/v1beta1', content_type='application/strategic-merge-patch+json'))
|
||||||
|
|
||||||
if data:
|
def _assert_success(self, response):
|
||||||
headers['Content-Type'] = content_type
|
if response.status_code != 200:
|
||||||
|
logger.error('Kubernetes API call failed with response: %s => %s', response.status_code,
|
||||||
|
response.text)
|
||||||
|
raise Exception('Kubernetes API call failed: %s' % response.text)
|
||||||
|
|
||||||
data = json.dumps(data) if data else None
|
def _update_secret_file(self, relative_file_path, value=None):
|
||||||
session = Session()
|
if '/' in relative_file_path:
|
||||||
url = 'https://%s/%s/%s' % (self.kube_config.api_host, api_prefix, relative_url)
|
raise Exception('Expected path from get_volume_path, but found slashes')
|
||||||
|
|
||||||
request = Request(method, url, data=data, headers=headers)
|
# Check first that the namespace for Quay Enterprise exists. If it does not, report that
|
||||||
return session.send(request.prepare(), verify=False, timeout=2)
|
# as an error, as it seems to be a common issue.
|
||||||
|
namespace_url = 'namespaces/%s' % (self.kube_config.qe_namespace)
|
||||||
|
response = self._execute_k8s_api('GET', namespace_url)
|
||||||
|
if response.status_code // 100 != 2:
|
||||||
|
msg = 'A Kubernetes namespace with name `%s` must be created to save config' % self.kube_config.qe_namespace
|
||||||
|
raise Exception(msg)
|
||||||
|
|
||||||
|
# Check if the secret exists. If not, then we create an empty secret and then update the file
|
||||||
|
# inside.
|
||||||
|
secret_url = 'namespaces/%s/secrets/%s' % (self.kube_config.qe_namespace, self.kube_config.qe_config_secret)
|
||||||
|
secret = self._lookup_secret()
|
||||||
|
if secret is None:
|
||||||
|
self._assert_success(self._execute_k8s_api('POST', secret_url, {
|
||||||
|
"kind": "Secret",
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"metadata": {
|
||||||
|
"name": self.kube_config.qe_config_secret
|
||||||
|
},
|
||||||
|
"data": {}
|
||||||
|
}))
|
||||||
|
|
||||||
|
# Update the secret to reflect the file change.
|
||||||
|
secret['data'] = secret.get('data', {})
|
||||||
|
|
||||||
|
if value is not None:
|
||||||
|
secret['data'][relative_file_path] = base64.b64encode(value)
|
||||||
|
else:
|
||||||
|
secret['data'].pop(relative_file_path)
|
||||||
|
|
||||||
|
self._assert_success(self._execute_k8s_api('PUT', secret_url, secret))
|
||||||
|
|
||||||
|
def _lookup_secret(self):
|
||||||
|
secret_url = 'namespaces/%s/secrets/%s' % (self.kube_config.qe_namespace, self.kube_config.qe_config_secret)
|
||||||
|
response = self._execute_k8s_api('GET', secret_url)
|
||||||
|
if response.status_code != 200:
|
||||||
|
return None
|
||||||
|
return json.loads(response.text)
|
||||||
|
|
||||||
|
def _execute_k8s_api(self, method, relative_url, data=None, api_prefix='api/v1', content_type='application/json'):
|
||||||
|
headers = {
|
||||||
|
'Authorization': 'Bearer ' + self.kube_config.service_account_token
|
||||||
|
}
|
||||||
|
|
||||||
|
if data:
|
||||||
|
headers['Content-Type'] = content_type
|
||||||
|
|
||||||
|
data = json.dumps(data) if data else None
|
||||||
|
session = Session()
|
||||||
|
url = 'https://%s/%s/%s' % (self.kube_config.api_host, api_prefix, relative_url)
|
||||||
|
|
||||||
|
request = Request(method, url, data=data, headers=headers)
|
||||||
|
return session.send(request.prepare(), verify=False, timeout=2)
|
||||||
|
|
|
@ -8,9 +8,11 @@ DEFAULT_QE_CONFIG_SECRET = 'quay-enterprise-config-secret'
|
||||||
# The name of the quay enterprise deployment (not config app) that is used to query & rollout
|
# The name of the quay enterprise deployment (not config app) that is used to query & rollout
|
||||||
DEFAULT_QE_DEPLOYMENT_SELECTOR = 'app'
|
DEFAULT_QE_DEPLOYMENT_SELECTOR = 'app'
|
||||||
|
|
||||||
|
|
||||||
def get_k8s_namespace():
|
def get_k8s_namespace():
|
||||||
return os.environ.get('QE_K8S_NAMESPACE', DEFAULT_QE_NAMESPACE)
|
return os.environ.get('QE_K8S_NAMESPACE', DEFAULT_QE_NAMESPACE)
|
||||||
|
|
||||||
|
|
||||||
class KubernetesConfig(object):
|
class KubernetesConfig(object):
|
||||||
def __init__(self, api_host='', service_account_token=SERVICE_ACCOUNT_TOKEN_PATH,
|
def __init__(self, api_host='', service_account_token=SERVICE_ACCOUNT_TOKEN_PATH,
|
||||||
qe_namespace=DEFAULT_QE_NAMESPACE,
|
qe_namespace=DEFAULT_QE_NAMESPACE,
|
||||||
|
@ -31,7 +33,7 @@ class KubernetesConfig(object):
|
||||||
with open(SERVICE_ACCOUNT_TOKEN_PATH, 'r') as f:
|
with open(SERVICE_ACCOUNT_TOKEN_PATH, 'r') as f:
|
||||||
service_token = f.read()
|
service_token = f.read()
|
||||||
|
|
||||||
api_host=os.environ.get('KUBERNETES_SERVICE_HOST', '')
|
api_host = os.environ.get('KUBERNETES_SERVICE_HOST', '')
|
||||||
port = os.environ.get('KUBERNETES_SERVICE_PORT')
|
port = os.environ.get('KUBERNETES_SERVICE_PORT')
|
||||||
if port:
|
if port:
|
||||||
api_host += ':' + port
|
api_host += ':' + port
|
||||||
|
@ -42,6 +44,3 @@ class KubernetesConfig(object):
|
||||||
|
|
||||||
return cls(api_host=api_host, service_account_token=service_token, qe_namespace=qe_namespace,
|
return cls(api_host=api_host, service_account_token=service_token, qe_namespace=qe_namespace,
|
||||||
qe_config_secret=qe_config_secret, qe_deployment_selector=qe_deployment_selector)
|
qe_config_secret=qe_config_secret, qe_deployment_selector=qe_deployment_selector)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -3,45 +3,45 @@ from config_app._init_config import CONF_DIR
|
||||||
|
|
||||||
|
|
||||||
def logfile_path(jsonfmt=False, debug=False):
|
def logfile_path(jsonfmt=False, debug=False):
|
||||||
"""
|
"""
|
||||||
Returns the a logfileconf path following this rules:
|
Returns the a logfileconf path following this rules:
|
||||||
- conf/logging_debug_json.conf # jsonfmt=true, debug=true
|
- conf/logging_debug_json.conf # jsonfmt=true, debug=true
|
||||||
- conf/logging_json.conf # jsonfmt=true, debug=false
|
- conf/logging_json.conf # jsonfmt=true, debug=false
|
||||||
- conf/logging_debug.conf # jsonfmt=false, debug=true
|
- conf/logging_debug.conf # jsonfmt=false, debug=true
|
||||||
- conf/logging.conf # jsonfmt=false, debug=false
|
- conf/logging.conf # jsonfmt=false, debug=false
|
||||||
Can be parametrized via envvars: JSONLOG=true, DEBUGLOG=true
|
Can be parametrized via envvars: JSONLOG=true, DEBUGLOG=true
|
||||||
"""
|
"""
|
||||||
_json = ""
|
_json = ""
|
||||||
_debug = ""
|
_debug = ""
|
||||||
|
|
||||||
if jsonfmt or os.getenv('JSONLOG', 'false').lower() == 'true':
|
if jsonfmt or os.getenv('JSONLOG', 'false').lower() == 'true':
|
||||||
_json = "_json"
|
_json = "_json"
|
||||||
|
|
||||||
if debug or os.getenv('DEBUGLOG', 'false').lower() == 'true':
|
if debug or os.getenv('DEBUGLOG', 'false').lower() == 'true':
|
||||||
_debug = "_debug"
|
_debug = "_debug"
|
||||||
|
|
||||||
return os.path.join(CONF_DIR, "logging%s%s.conf" % (_debug, _json))
|
return os.path.join(CONF_DIR, "logging%s%s.conf" % (_debug, _json))
|
||||||
|
|
||||||
|
|
||||||
def filter_logs(values, filtered_fields):
|
def filter_logs(values, filtered_fields):
|
||||||
"""
|
"""
|
||||||
Takes a dict and a list of keys to filter.
|
Takes a dict and a list of keys to filter.
|
||||||
eg:
|
eg:
|
||||||
with filtered_fields:
|
with filtered_fields:
|
||||||
[{'key': ['k1', k2'], 'fn': lambda x: 'filtered'}]
|
[{'key': ['k1', k2'], 'fn': lambda x: 'filtered'}]
|
||||||
and values:
|
and values:
|
||||||
{'k1': {'k2': 'some-secret'}, 'k3': 'some-value'}
|
{'k1': {'k2': 'some-secret'}, 'k3': 'some-value'}
|
||||||
the returned dict is:
|
the returned dict is:
|
||||||
{'k1': {k2: 'filtered'}, 'k3': 'some-value'}
|
{'k1': {k2: 'filtered'}, 'k3': 'some-value'}
|
||||||
"""
|
"""
|
||||||
for field in filtered_fields:
|
for field in filtered_fields:
|
||||||
cdict = values
|
cdict = values
|
||||||
|
|
||||||
for key in field['key'][:-1]:
|
for key in field['key'][:-1]:
|
||||||
if key in cdict:
|
if key in cdict:
|
||||||
cdict = cdict[key]
|
cdict = cdict[key]
|
||||||
|
|
||||||
last_key = field['key'][-1]
|
last_key = field['key'][-1]
|
||||||
|
|
||||||
if last_key in cdict and cdict[last_key]:
|
if last_key in cdict and cdict[last_key]:
|
||||||
cdict[last_key] = field['fn'](cdict[last_key])
|
cdict[last_key] = field['fn'](cdict[last_key])
|
||||||
|
|
|
@ -2,10 +2,12 @@ from fnmatch import fnmatch
|
||||||
|
|
||||||
import OpenSSL
|
import OpenSSL
|
||||||
|
|
||||||
|
|
||||||
class CertInvalidException(Exception):
|
class CertInvalidException(Exception):
|
||||||
""" Exception raised when a certificate could not be parsed/loaded. """
|
""" Exception raised when a certificate could not be parsed/loaded. """
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
class KeyInvalidException(Exception):
|
class KeyInvalidException(Exception):
|
||||||
""" Exception raised when a key could not be parsed/loaded or successfully applied to a cert. """
|
""" Exception raised when a key could not be parsed/loaded or successfully applied to a cert. """
|
||||||
pass
|
pass
|
||||||
|
@ -24,8 +26,10 @@ def load_certificate(cert_contents):
|
||||||
|
|
||||||
_SUBJECT_ALT_NAME = 'subjectAltName'
|
_SUBJECT_ALT_NAME = 'subjectAltName'
|
||||||
|
|
||||||
|
|
||||||
class SSLCertificate(object):
|
class SSLCertificate(object):
|
||||||
""" Helper class for easier working with SSL certificates. """
|
""" Helper class for easier working with SSL certificates. """
|
||||||
|
|
||||||
def __init__(self, openssl_cert):
|
def __init__(self, openssl_cert):
|
||||||
self.openssl_cert = openssl_cert
|
self.openssl_cert = openssl_cert
|
||||||
|
|
||||||
|
|
|
@ -1,20 +1,22 @@
|
||||||
from util.config.validator import EXTRA_CA_DIRECTORY
|
from util.config.validator import EXTRA_CA_DIRECTORY
|
||||||
|
|
||||||
|
|
||||||
def strip_absolute_path_and_add_trailing_dir(path):
|
def strip_absolute_path_and_add_trailing_dir(path):
|
||||||
"""
|
"""
|
||||||
Removes the initial trailing / from the prefix path, and add the last dir one
|
Removes the initial trailing / from the prefix path, and add the last dir one
|
||||||
"""
|
"""
|
||||||
return path[1:] + '/'
|
return path[1:] + '/'
|
||||||
|
|
||||||
|
|
||||||
def tarinfo_filter_partial(prefix):
|
def tarinfo_filter_partial(prefix):
|
||||||
def tarinfo_filter(tarinfo):
|
def tarinfo_filter(tarinfo):
|
||||||
# remove leading directory info
|
# remove leading directory info
|
||||||
tarinfo.name = tarinfo.name.replace(prefix, '')
|
tarinfo.name = tarinfo.name.replace(prefix, '')
|
||||||
|
|
||||||
# ignore any directory that isn't the specified extra ca one:
|
# ignore any directory that isn't the specified extra ca one:
|
||||||
if tarinfo.isdir() and not tarinfo.name == EXTRA_CA_DIRECTORY:
|
if tarinfo.isdir() and not tarinfo.name == EXTRA_CA_DIRECTORY:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
return tarinfo
|
return tarinfo
|
||||||
|
|
||||||
return tarinfo_filter
|
return tarinfo_filter
|
||||||
|
|
|
@ -1,23 +1,25 @@
|
||||||
import pytest
|
import pytest
|
||||||
import re
|
|
||||||
|
|
||||||
from httmock import urlmatch, HTTMock, response
|
from httmock import urlmatch, HTTMock, response
|
||||||
|
|
||||||
from config_app.config_util.k8saccessor import KubernetesAccessorSingleton
|
from config_app.config_util.k8saccessor import KubernetesAccessorSingleton
|
||||||
from config_app.config_util.k8sconfig import KubernetesConfig
|
from config_app.config_util.k8sconfig import KubernetesConfig
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize('kube_config, expected_api, expected_query', [
|
@pytest.mark.parametrize('kube_config, expected_api, expected_query', [
|
||||||
({'api_host':'www.customhost.com'},
|
({'api_host': 'www.customhost.com'},
|
||||||
'/apis/extensions/v1beta1/namespaces/quay-enterprise/deployments', 'labelSelector=quay-enterprise-component%3Dapp'),
|
'/apis/extensions/v1beta1/namespaces/quay-enterprise/deployments', 'labelSelector=quay-enterprise-component%3Dapp'),
|
||||||
|
|
||||||
({'api_host':'www.customhost.com', 'qe_deployment_selector':'custom-selector'},
|
({'api_host': 'www.customhost.com', 'qe_deployment_selector': 'custom-selector'},
|
||||||
'/apis/extensions/v1beta1/namespaces/quay-enterprise/deployments', 'labelSelector=quay-enterprise-component%3Dcustom-selector'),
|
'/apis/extensions/v1beta1/namespaces/quay-enterprise/deployments',
|
||||||
|
'labelSelector=quay-enterprise-component%3Dcustom-selector'),
|
||||||
|
|
||||||
({'api_host':'www.customhost.com', 'qe_namespace':'custom-namespace'},
|
({'api_host': 'www.customhost.com', 'qe_namespace': 'custom-namespace'},
|
||||||
'/apis/extensions/v1beta1/namespaces/custom-namespace/deployments', 'labelSelector=quay-enterprise-component%3Dapp'),
|
'/apis/extensions/v1beta1/namespaces/custom-namespace/deployments', 'labelSelector=quay-enterprise-component%3Dapp'),
|
||||||
|
|
||||||
({'api_host':'www.customhost.com', 'qe_namespace':'custom-namespace', 'qe_deployment_selector':'custom-selector'},
|
({'api_host': 'www.customhost.com', 'qe_namespace': 'custom-namespace', 'qe_deployment_selector': 'custom-selector'},
|
||||||
'/apis/extensions/v1beta1/namespaces/custom-namespace/deployments', 'labelSelector=quay-enterprise-component%3Dcustom-selector'),
|
'/apis/extensions/v1beta1/namespaces/custom-namespace/deployments',
|
||||||
|
'labelSelector=quay-enterprise-component%3Dcustom-selector'),
|
||||||
])
|
])
|
||||||
def test_get_qe_deployments(kube_config, expected_api, expected_query):
|
def test_get_qe_deployments(kube_config, expected_api, expected_query):
|
||||||
config = KubernetesConfig(**kube_config)
|
config = KubernetesConfig(**kube_config)
|
||||||
|
@ -36,12 +38,15 @@ def test_get_qe_deployments(kube_config, expected_api, expected_query):
|
||||||
|
|
||||||
assert url_hit[0]
|
assert url_hit[0]
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize('kube_config, deployment_names, expected_api_hits', [
|
@pytest.mark.parametrize('kube_config, deployment_names, expected_api_hits', [
|
||||||
({'api_host':'www.customhost.com'}, [], []),
|
({'api_host': 'www.customhost.com'}, [], []),
|
||||||
({'api_host':'www.customhost.com'}, ['myDeployment'], ['/apis/extensions/v1beta1/namespaces/quay-enterprise/deployments/myDeployment']),
|
({'api_host': 'www.customhost.com'}, ['myDeployment'],
|
||||||
({'api_host':'www.customhost.com', 'qe_namespace':'custom-namespace'},
|
['/apis/extensions/v1beta1/namespaces/quay-enterprise/deployments/myDeployment']),
|
||||||
|
({'api_host': 'www.customhost.com', 'qe_namespace': 'custom-namespace'},
|
||||||
['myDeployment', 'otherDeployment'],
|
['myDeployment', 'otherDeployment'],
|
||||||
['/apis/extensions/v1beta1/namespaces/custom-namespace/deployments/myDeployment', '/apis/extensions/v1beta1/namespaces/custom-namespace/deployments/otherDeployment']),
|
['/apis/extensions/v1beta1/namespaces/custom-namespace/deployments/myDeployment',
|
||||||
|
'/apis/extensions/v1beta1/namespaces/custom-namespace/deployments/otherDeployment']),
|
||||||
])
|
])
|
||||||
def test_cycle_qe_deployments(kube_config, deployment_names, expected_api_hits):
|
def test_cycle_qe_deployments(kube_config, deployment_names, expected_api_hits):
|
||||||
KubernetesAccessorSingleton._instance = None
|
KubernetesAccessorSingleton._instance = None
|
||||||
|
|
|
@ -6,24 +6,27 @@ from util.config.validator import EXTRA_CA_DIRECTORY
|
||||||
|
|
||||||
from test.fixtures import *
|
from test.fixtures import *
|
||||||
|
|
||||||
class MockTarInfo:
|
|
||||||
def __init__(self, name, isdir):
|
|
||||||
self.name = name
|
|
||||||
self.isdir = lambda: isdir
|
|
||||||
|
|
||||||
def __eq__(self, other):
|
class MockTarInfo:
|
||||||
return other is not None and self.name == other.name
|
def __init__(self, name, isdir):
|
||||||
|
self.name = name
|
||||||
|
self.isdir = lambda: isdir
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
return other is not None and self.name == other.name
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize('prefix,tarinfo,expected', [
|
@pytest.mark.parametrize('prefix,tarinfo,expected', [
|
||||||
# It should handle simple files
|
# It should handle simple files
|
||||||
('Users/sam/', MockTarInfo('Users/sam/config.yaml', False), MockTarInfo('config.yaml', False)),
|
('Users/sam/', MockTarInfo('Users/sam/config.yaml', False), MockTarInfo('config.yaml', False)),
|
||||||
# It should allow the extra CA dir
|
# It should allow the extra CA dir
|
||||||
('Users/sam/', MockTarInfo('Users/sam/%s' % EXTRA_CA_DIRECTORY, True), MockTarInfo('%s' % EXTRA_CA_DIRECTORY, True)),
|
('Users/sam/', MockTarInfo('Users/sam/%s' % EXTRA_CA_DIRECTORY, True), MockTarInfo('%s' % EXTRA_CA_DIRECTORY, True)),
|
||||||
# it should allow a file in that extra dir
|
# it should allow a file in that extra dir
|
||||||
('Users/sam/', MockTarInfo('Users/sam/%s/cert.crt' % EXTRA_CA_DIRECTORY, False), MockTarInfo('%s/cert.crt' % EXTRA_CA_DIRECTORY, False)),
|
('Users/sam/', MockTarInfo('Users/sam/%s/cert.crt' % EXTRA_CA_DIRECTORY, False),
|
||||||
# it should not allow a directory that isn't the CA dir
|
MockTarInfo('%s/cert.crt' % EXTRA_CA_DIRECTORY, False)),
|
||||||
('Users/sam/', MockTarInfo('Users/sam/dirignore', True), None),
|
# it should not allow a directory that isn't the CA dir
|
||||||
|
('Users/sam/', MockTarInfo('Users/sam/dirignore', True), None),
|
||||||
])
|
])
|
||||||
def test_tarinfo_filter(prefix, tarinfo, expected):
|
def test_tarinfo_filter(prefix, tarinfo, expected):
|
||||||
partial = tarinfo_filter_partial(prefix)
|
partial = tarinfo_filter_partial(prefix)
|
||||||
assert partial(tarinfo) == expected
|
assert partial(tarinfo) == expected
|
||||||
|
|
|
@ -2,7 +2,5 @@ from config_app.c_app import app as application
|
||||||
from config_app.config_endpoints.api import api_bp
|
from config_app.config_endpoints.api import api_bp
|
||||||
from config_app.config_endpoints.setup_web import setup_web
|
from config_app.config_endpoints.setup_web import setup_web
|
||||||
|
|
||||||
|
|
||||||
application.register_blueprint(setup_web)
|
application.register_blueprint(setup_web)
|
||||||
application.register_blueprint(api_bp, url_prefix='/api')
|
application.register_blueprint(api_bp, url_prefix='/api')
|
||||||
|
|
||||||
|
|
Reference in a new issue