2013-09-20 15:55:44 +00:00
|
|
|
import json
|
|
|
|
import logging
|
2013-10-29 20:11:54 +00:00
|
|
|
import urlparse
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2015-07-15 21:25:41 +00:00
|
|
|
from flask import request, make_response, jsonify, session
|
2013-09-20 15:55:44 +00:00
|
|
|
from functools import wraps
|
|
|
|
|
2014-03-18 23:21:27 +00:00
|
|
|
from data import model
|
2014-10-29 19:42:44 +00:00
|
|
|
from app import app, authentication, userevents, storage
|
2015-02-19 21:54:23 +00:00
|
|
|
from auth.auth import process_auth, generate_signed_token
|
2014-03-26 20:19:04 +00:00
|
|
|
from auth.auth_context import get_authenticated_user, get_validated_token, get_validated_oauth_token
|
2013-11-11 21:28:05 +00:00
|
|
|
from util.names import parse_repository_name
|
2014-03-18 23:21:27 +00:00
|
|
|
from auth.permissions import (ModifyRepositoryPermission, UserAdminPermission,
|
2015-02-19 21:54:23 +00:00
|
|
|
ReadRepositoryPermission, CreateRepositoryPermission,
|
2015-07-15 21:25:41 +00:00
|
|
|
repository_read_grant, repository_write_grant)
|
2014-01-28 23:29:45 +00:00
|
|
|
|
2014-01-24 20:01:40 +00:00
|
|
|
from util.http import abort
|
2015-06-22 21:37:13 +00:00
|
|
|
from endpoints.v1 import v1_bp
|
2014-10-30 16:49:51 +00:00
|
|
|
from endpoints.trackhelper import track_and_log
|
2014-07-18 20:34:52 +00:00
|
|
|
from endpoints.notificationhelper import spawn_notification
|
2015-06-02 19:56:44 +00:00
|
|
|
from endpoints.decorators import anon_protect, anon_allowed
|
2014-01-24 20:01:40 +00:00
|
|
|
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
logger = logging.getLogger(__name__)
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2015-02-19 21:54:23 +00:00
|
|
|
|
|
|
|
class GrantType(object):
|
|
|
|
READ_REPOSITORY = 'read'
|
|
|
|
WRITE_REPOSITORY = 'write'
|
|
|
|
|
|
|
|
|
2015-06-10 19:16:01 +00:00
|
|
|
def generate_headers(scope=GrantType.READ_REPOSITORY, add_grant_for_status=None):
|
2013-10-16 18:24:10 +00:00
|
|
|
def decorator_method(f):
|
|
|
|
@wraps(f)
|
|
|
|
def wrapper(namespace, repository, *args, **kwargs):
|
|
|
|
response = f(namespace, repository, *args, **kwargs)
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2013-12-09 04:24:29 +00:00
|
|
|
# Setting session namespace and repository
|
|
|
|
session['namespace'] = namespace
|
|
|
|
session['repository'] = repository
|
2015-06-02 19:16:22 +00:00
|
|
|
|
2013-10-29 20:11:54 +00:00
|
|
|
# We run our index and registry on the same hosts for now
|
|
|
|
registry_server = urlparse.urlparse(request.url).netloc
|
|
|
|
response.headers['X-Docker-Endpoints'] = registry_server
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2013-10-16 18:24:10 +00:00
|
|
|
has_token_request = request.headers.get('X-Docker-Token', '')
|
2015-06-10 19:16:01 +00:00
|
|
|
force_grant = (add_grant_for_status == response.status_code)
|
2013-09-20 22:38:17 +00:00
|
|
|
|
2015-06-10 19:16:01 +00:00
|
|
|
if has_token_request or force_grant:
|
2015-02-19 21:54:23 +00:00
|
|
|
grants = []
|
2015-06-02 19:16:22 +00:00
|
|
|
|
2015-02-19 21:54:23 +00:00
|
|
|
if scope == GrantType.READ_REPOSITORY:
|
2015-06-19 18:02:51 +00:00
|
|
|
if force_grant or ReadRepositoryPermission(namespace, repository).can():
|
2015-06-02 19:16:22 +00:00
|
|
|
grants.append(repository_read_grant(namespace, repository))
|
2015-02-19 21:54:23 +00:00
|
|
|
elif scope == GrantType.WRITE_REPOSITORY:
|
2015-06-19 18:02:51 +00:00
|
|
|
if force_grant or ModifyRepositoryPermission(namespace, repository).can():
|
2015-06-02 19:16:22 +00:00
|
|
|
grants.append(repository_write_grant(namespace, repository))
|
2015-02-19 21:54:23 +00:00
|
|
|
|
2015-06-02 19:16:22 +00:00
|
|
|
# Generate a signed token for the user (if any) and the grants (if any)
|
|
|
|
if grants or get_authenticated_user():
|
2015-02-24 18:22:19 +00:00
|
|
|
user_context = get_authenticated_user() and get_authenticated_user().username
|
|
|
|
signature = generate_signed_token(grants, user_context)
|
|
|
|
response.headers['WWW-Authenticate'] = signature
|
|
|
|
response.headers['X-Docker-Token'] = signature
|
2015-06-02 19:16:22 +00:00
|
|
|
|
2013-10-16 18:24:10 +00:00
|
|
|
return response
|
|
|
|
return wrapper
|
|
|
|
return decorator_method
|
2013-09-20 15:55:44 +00:00
|
|
|
|
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
@v1_bp.route('/users', methods=['POST'])
|
|
|
|
@v1_bp.route('/users/', methods=['POST'])
|
2015-06-02 19:56:44 +00:00
|
|
|
@anon_allowed
|
2013-09-20 15:55:44 +00:00
|
|
|
def create_user():
|
|
|
|
user_data = request.get_json()
|
2014-10-14 19:46:35 +00:00
|
|
|
if not user_data or not 'username' in user_data:
|
2014-09-15 15:27:33 +00:00
|
|
|
abort(400, 'Missing username')
|
|
|
|
|
2013-10-01 16:25:06 +00:00
|
|
|
username = user_data['username']
|
2014-07-15 01:24:38 +00:00
|
|
|
password = user_data.get('password', '')
|
2013-10-01 16:25:06 +00:00
|
|
|
|
2014-02-20 19:49:34 +00:00
|
|
|
# UGH! we have to use this response when the login actually worked, in order
|
|
|
|
# to get the CLI to try again with a get, and then tell us login succeeded.
|
|
|
|
success = make_response('"Username or email already exists"', 400)
|
|
|
|
|
2013-10-16 18:24:10 +00:00
|
|
|
if username == '$token':
|
|
|
|
try:
|
2015-07-15 21:25:41 +00:00
|
|
|
model.token.load_token_data(password)
|
2014-02-20 19:49:34 +00:00
|
|
|
return success
|
2013-10-16 18:24:10 +00:00
|
|
|
except model.InvalidTokenException:
|
2014-01-28 23:29:45 +00:00
|
|
|
abort(400, 'Invalid access token.', issue='invalid-access-token')
|
2013-10-16 18:24:10 +00:00
|
|
|
|
2014-03-20 16:09:25 +00:00
|
|
|
elif username == '$oauthtoken':
|
2015-07-15 21:25:41 +00:00
|
|
|
validated = model.oauth.validate_access_token(password)
|
2014-03-20 16:09:25 +00:00
|
|
|
if validated is not None:
|
|
|
|
return success
|
|
|
|
else:
|
|
|
|
abort(400, 'Invalid oauth access token.', issue='invalid-oauth-access-token')
|
|
|
|
|
2013-11-21 00:43:19 +00:00
|
|
|
elif '+' in username:
|
|
|
|
try:
|
2015-07-15 21:25:41 +00:00
|
|
|
model.user.verify_robot(username, password)
|
2014-02-20 19:49:34 +00:00
|
|
|
return success
|
2013-11-21 00:43:19 +00:00
|
|
|
except model.InvalidRobotException:
|
2014-02-20 19:49:34 +00:00
|
|
|
abort(400, 'Invalid robot account or password.',
|
|
|
|
issue='robot-login-failure')
|
2013-11-21 00:43:19 +00:00
|
|
|
|
2015-07-20 15:39:59 +00:00
|
|
|
(verified, error_message) = authentication.verify_and_link_user(username, password,
|
|
|
|
basic_auth=True)
|
2015-06-22 21:12:05 +00:00
|
|
|
if verified:
|
|
|
|
# Mark that the user was logged in.
|
|
|
|
event = userevents.get_event(username)
|
|
|
|
event.publish_event_data('docker-cli', {'action': 'login'})
|
|
|
|
return success
|
2013-10-01 16:25:06 +00:00
|
|
|
else:
|
2015-06-22 21:12:05 +00:00
|
|
|
# Mark that the login failed.
|
|
|
|
event = userevents.get_event(username)
|
|
|
|
event.publish_event_data('docker-cli', {'action': 'loginfailure'})
|
|
|
|
abort(400, error_message, issue='login-failure')
|
2013-09-20 15:55:44 +00:00
|
|
|
|
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
@v1_bp.route('/users', methods=['GET'])
|
|
|
|
@v1_bp.route('/users/', methods=['GET'])
|
2013-09-20 22:38:17 +00:00
|
|
|
@process_auth
|
2015-06-02 19:56:44 +00:00
|
|
|
@anon_allowed
|
2013-09-20 15:55:44 +00:00
|
|
|
def get_user():
|
2014-03-26 20:19:04 +00:00
|
|
|
if get_validated_oauth_token():
|
|
|
|
return jsonify({
|
|
|
|
'username': '$oauthtoken',
|
|
|
|
'email': None,
|
|
|
|
})
|
2014-11-24 21:07:38 +00:00
|
|
|
elif get_authenticated_user():
|
2013-11-07 22:09:47 +00:00
|
|
|
return jsonify({
|
|
|
|
'username': get_authenticated_user().username,
|
|
|
|
'email': get_authenticated_user().email,
|
|
|
|
})
|
2014-02-21 21:07:08 +00:00
|
|
|
elif get_validated_token():
|
|
|
|
return jsonify({
|
|
|
|
'username': '$token',
|
|
|
|
'email': None,
|
|
|
|
})
|
2013-11-07 22:09:47 +00:00
|
|
|
abort(404)
|
2013-09-20 15:55:44 +00:00
|
|
|
|
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
@v1_bp.route('/users/<username>/', methods=['PUT'])
|
2013-09-20 22:38:17 +00:00
|
|
|
@process_auth
|
2015-06-02 19:56:44 +00:00
|
|
|
@anon_allowed
|
2013-09-20 15:55:44 +00:00
|
|
|
def update_user(username):
|
2014-03-18 23:21:27 +00:00
|
|
|
permission = UserAdminPermission(username)
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2013-09-20 22:38:17 +00:00
|
|
|
if permission.can():
|
|
|
|
update_request = request.get_json()
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2013-09-20 22:38:17 +00:00
|
|
|
if 'password' in update_request:
|
2015-02-11 19:15:18 +00:00
|
|
|
logger.debug('Updating user password')
|
2015-07-15 21:25:41 +00:00
|
|
|
model.user.change_password(get_authenticated_user(), update_request['password'])
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2013-09-20 22:38:17 +00:00
|
|
|
if 'email' in update_request:
|
2015-02-11 19:15:18 +00:00
|
|
|
logger.debug('Updating user email')
|
2015-07-15 21:25:41 +00:00
|
|
|
model.user.update_email(get_authenticated_user(), update_request['email'])
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2013-09-20 22:38:17 +00:00
|
|
|
return jsonify({
|
|
|
|
'username': get_authenticated_user().username,
|
|
|
|
'email': get_authenticated_user().email,
|
|
|
|
})
|
|
|
|
|
|
|
|
abort(403)
|
2013-09-20 15:55:44 +00:00
|
|
|
|
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
@v1_bp.route('/repositories/<path:repository>', methods=['PUT'])
|
2013-09-20 22:38:17 +00:00
|
|
|
@process_auth
|
2013-09-20 15:55:44 +00:00
|
|
|
@parse_repository_name
|
2015-06-10 19:16:01 +00:00
|
|
|
@generate_headers(scope=GrantType.WRITE_REPOSITORY, add_grant_for_status=201)
|
2015-06-02 19:56:44 +00:00
|
|
|
@anon_allowed
|
2013-09-20 15:55:44 +00:00
|
|
|
def create_repository(namespace, repository):
|
2015-06-10 19:16:01 +00:00
|
|
|
logger.debug('Looking up repository %s/%s', namespace, repository)
|
2015-07-15 21:25:41 +00:00
|
|
|
repo = model.repository.get_repository(namespace, repository)
|
2013-09-20 22:38:17 +00:00
|
|
|
|
2015-06-10 19:16:01 +00:00
|
|
|
logger.debug('Found repository %s/%s', namespace, repository)
|
2013-10-16 18:24:10 +00:00
|
|
|
if not repo and get_authenticated_user() is None:
|
2015-06-10 19:16:01 +00:00
|
|
|
logger.debug('Attempt to create repository %s/%s without user auth', namespace, repository)
|
2014-01-28 23:29:45 +00:00
|
|
|
abort(401,
|
|
|
|
message='Cannot create a repository as a guest. Please login via "docker login" first.',
|
|
|
|
issue='no-login')
|
2013-10-16 18:24:10 +00:00
|
|
|
|
|
|
|
elif repo:
|
2013-09-20 22:38:17 +00:00
|
|
|
permission = ModifyRepositoryPermission(namespace, repository)
|
|
|
|
if not permission.can():
|
2014-11-24 21:07:38 +00:00
|
|
|
abort(403,
|
2014-01-28 23:29:45 +00:00
|
|
|
message='You do not have permission to modify repository %(namespace)s/%(repository)s',
|
|
|
|
issue='no-repo-write-permission',
|
|
|
|
namespace=namespace, repository=repository)
|
2013-09-26 19:58:11 +00:00
|
|
|
else:
|
2013-11-11 21:31:29 +00:00
|
|
|
permission = CreateRepositoryPermission(namespace)
|
2013-11-04 20:42:08 +00:00
|
|
|
if not permission.can():
|
2015-07-15 21:25:41 +00:00
|
|
|
logger.info('Attempt to create a new repo %s/%s with insufficient perms', namespace,
|
|
|
|
repository)
|
|
|
|
msg = 'You do not have permission to create repositories in namespace "%(namespace)s"'
|
|
|
|
abort(403, message=msg, issue='no-create-permission', namespace=namespace)
|
2013-09-20 22:38:17 +00:00
|
|
|
|
2015-06-10 19:16:01 +00:00
|
|
|
# Attempt to create the new repository.
|
|
|
|
logger.debug('Creating repository %s/%s with owner: %s', namespace, repository,
|
|
|
|
get_authenticated_user().username)
|
|
|
|
|
2015-07-15 21:25:41 +00:00
|
|
|
repo = model.repository.create_repository(namespace, repository, get_authenticated_user())
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2015-05-13 18:55:39 +00:00
|
|
|
if get_authenticated_user():
|
|
|
|
user_event_data = {
|
|
|
|
'action': 'push_start',
|
|
|
|
'repository': repository,
|
|
|
|
'namespace': namespace
|
|
|
|
}
|
|
|
|
|
|
|
|
event = userevents.get_event(get_authenticated_user().username)
|
|
|
|
event.publish_event_data('docker-cli', user_event_data)
|
2015-04-03 16:13:33 +00:00
|
|
|
|
2014-11-06 19:48:16 +00:00
|
|
|
return make_response('Created', 201)
|
2013-09-20 15:55:44 +00:00
|
|
|
|
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
@v1_bp.route('/repositories/<path:repository>/images', methods=['PUT'])
|
2013-09-20 22:38:17 +00:00
|
|
|
@process_auth
|
2013-09-20 15:55:44 +00:00
|
|
|
@parse_repository_name
|
2015-02-19 21:54:23 +00:00
|
|
|
@generate_headers(scope=GrantType.WRITE_REPOSITORY)
|
2015-06-02 19:56:44 +00:00
|
|
|
@anon_allowed
|
2013-09-20 15:55:44 +00:00
|
|
|
def update_images(namespace, repository):
|
2013-09-20 22:38:17 +00:00
|
|
|
permission = ModifyRepositoryPermission(namespace, repository)
|
|
|
|
|
|
|
|
if permission.can():
|
2015-02-11 19:15:18 +00:00
|
|
|
logger.debug('Looking up repository')
|
2015-07-15 21:25:41 +00:00
|
|
|
repo = model.repository.get_repository(namespace, repository)
|
2013-11-16 20:05:26 +00:00
|
|
|
if not repo:
|
2013-11-11 23:05:21 +00:00
|
|
|
# Make sure the repo actually exists.
|
2014-01-28 23:29:45 +00:00
|
|
|
abort(404, message='Unknown repository', issue='unknown-repo')
|
2013-11-11 23:05:21 +00:00
|
|
|
|
2015-02-11 19:15:18 +00:00
|
|
|
logger.debug('GCing repository')
|
2015-07-15 21:25:41 +00:00
|
|
|
model.repository.garbage_collect_repository(namespace, repository)
|
2014-02-06 19:13:35 +00:00
|
|
|
|
2014-07-18 02:51:58 +00:00
|
|
|
# Generate a job for each notification that has been added to this repo
|
2015-02-11 19:15:18 +00:00
|
|
|
logger.debug('Adding notifications for repository')
|
2014-07-18 02:51:58 +00:00
|
|
|
|
2014-10-22 18:14:56 +00:00
|
|
|
updated_tags = session.get('pushed_tags', {})
|
2014-07-18 02:51:58 +00:00
|
|
|
event_data = {
|
|
|
|
'updated_tags': updated_tags,
|
|
|
|
}
|
2015-04-03 16:13:33 +00:00
|
|
|
|
2015-02-13 21:28:45 +00:00
|
|
|
track_and_log('push_repo', repo)
|
2014-07-18 19:58:18 +00:00
|
|
|
spawn_notification(repo, 'repo_push', event_data)
|
2013-09-20 22:38:17 +00:00
|
|
|
return make_response('Updated', 204)
|
|
|
|
|
|
|
|
abort(403)
|
2013-09-20 15:55:44 +00:00
|
|
|
|
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
@v1_bp.route('/repositories/<path:repository>/images', methods=['GET'])
|
2013-09-20 22:38:17 +00:00
|
|
|
@process_auth
|
2013-09-20 15:55:44 +00:00
|
|
|
@parse_repository_name
|
2015-02-19 21:54:23 +00:00
|
|
|
@generate_headers(scope=GrantType.READ_REPOSITORY)
|
2015-05-19 21:52:44 +00:00
|
|
|
@anon_protect
|
2013-09-20 15:55:44 +00:00
|
|
|
def get_repository_images(namespace, repository):
|
2013-09-20 22:38:17 +00:00
|
|
|
permission = ReadRepositoryPermission(namespace, repository)
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2013-09-20 22:38:17 +00:00
|
|
|
# TODO invalidate token?
|
2015-07-15 21:25:41 +00:00
|
|
|
if permission.can() or model.repository.repository_is_public(namespace, repository):
|
2013-11-11 23:05:21 +00:00
|
|
|
# We can't rely on permissions to tell us if a repo exists anymore
|
2015-02-11 19:15:18 +00:00
|
|
|
logger.debug('Looking up repository')
|
2015-07-15 21:25:41 +00:00
|
|
|
repo = model.repository.get_repository(namespace, repository)
|
2013-11-11 23:05:21 +00:00
|
|
|
if not repo:
|
2014-01-28 23:29:45 +00:00
|
|
|
abort(404, message='Unknown repository', issue='unknown-repo')
|
2013-11-11 23:05:21 +00:00
|
|
|
|
2015-02-11 19:15:18 +00:00
|
|
|
logger.debug('Building repository image response')
|
2015-03-04 21:36:32 +00:00
|
|
|
resp = make_response(json.dumps([]), 200)
|
2013-09-20 22:38:17 +00:00
|
|
|
resp.mimetype = 'application/json'
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2015-09-15 19:23:06 +00:00
|
|
|
track_and_log('pull_repo', repo, analytics_name='pull_repo_100x', analytics_sample=0.01)
|
2013-09-20 22:38:17 +00:00
|
|
|
return resp
|
2013-09-20 15:55:44 +00:00
|
|
|
|
2013-11-11 23:05:21 +00:00
|
|
|
abort(403)
|
2013-09-20 15:55:44 +00:00
|
|
|
|
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
@v1_bp.route('/repositories/<path:repository>/images', methods=['DELETE'])
|
2013-09-20 22:38:17 +00:00
|
|
|
@process_auth
|
2013-09-20 15:55:44 +00:00
|
|
|
@parse_repository_name
|
2015-02-19 21:54:23 +00:00
|
|
|
@generate_headers(scope=GrantType.WRITE_REPOSITORY)
|
2015-06-02 19:56:44 +00:00
|
|
|
@anon_allowed
|
2013-09-20 15:55:44 +00:00
|
|
|
def delete_repository_images(namespace, repository):
|
2014-01-28 23:29:45 +00:00
|
|
|
abort(501, 'Not Implemented', issue='not-implemented')
|
2013-09-20 15:55:44 +00:00
|
|
|
|
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
@v1_bp.route('/repositories/<path:repository>/auth', methods=['PUT'])
|
2013-09-20 15:55:44 +00:00
|
|
|
@parse_repository_name
|
2015-06-02 19:56:44 +00:00
|
|
|
@anon_allowed
|
2013-09-20 15:55:44 +00:00
|
|
|
def put_repository_auth(namespace, repository):
|
2014-01-28 23:29:45 +00:00
|
|
|
abort(501, 'Not Implemented', issue='not-implemented')
|
2013-09-20 15:55:44 +00:00
|
|
|
|
|
|
|
|
2015-06-22 21:37:13 +00:00
|
|
|
@v1_bp.route('/search', methods=['GET'])
|
2014-08-22 23:41:22 +00:00
|
|
|
@process_auth
|
2015-05-19 21:52:44 +00:00
|
|
|
@anon_protect
|
2013-09-20 15:55:44 +00:00
|
|
|
def get_search():
|
2014-08-22 23:41:22 +00:00
|
|
|
def result_view(repo):
|
|
|
|
return {
|
2014-09-24 22:01:35 +00:00
|
|
|
"name": repo.namespace_user.username + '/' + repo.name,
|
2014-08-22 23:41:22 +00:00
|
|
|
"description": repo.description
|
|
|
|
}
|
|
|
|
|
|
|
|
query = request.args.get('q')
|
|
|
|
|
|
|
|
username = None
|
|
|
|
user = get_authenticated_user()
|
|
|
|
if user is not None:
|
|
|
|
username = user.username
|
|
|
|
|
2014-08-25 18:23:21 +00:00
|
|
|
if query:
|
2015-07-15 21:25:41 +00:00
|
|
|
matching = model.repository.get_matching_repositories(query, username)
|
2014-08-25 18:23:21 +00:00
|
|
|
else:
|
|
|
|
matching = []
|
|
|
|
|
2014-08-22 23:41:22 +00:00
|
|
|
results = [result_view(repo) for repo in matching
|
2014-11-24 21:07:38 +00:00
|
|
|
if (repo.visibility.name == 'public' or
|
2014-09-24 22:01:35 +00:00
|
|
|
ReadRepositoryPermission(repo.namespace_user.username, repo.name).can())]
|
2014-08-22 23:41:22 +00:00
|
|
|
|
|
|
|
data = {
|
|
|
|
"query": query,
|
|
|
|
"num_results": len(results),
|
|
|
|
"results" : results
|
|
|
|
}
|
|
|
|
|
|
|
|
resp = make_response(json.dumps(data), 200)
|
|
|
|
resp.mimetype = 'application/json'
|
|
|
|
return resp
|