import logging
import json

from flask import abort, request, jsonify, make_response, session

from app import app
from util.names import TAG_ERROR, TAG_REGEX, parse_repository_name
from auth.auth import process_auth
from auth.permissions import (ReadRepositoryPermission,
                              ModifyRepositoryPermission)
from data import model
from endpoints.decorators import anon_protect
from endpoints.v1 import v1_bp


logger = logging.getLogger(__name__)


@v1_bp.route('/repositories/<path:repository>/tags', methods=['GET'])
@process_auth
@anon_protect
@parse_repository_name
def get_tags(namespace, repository):
  permission = ReadRepositoryPermission(namespace, repository)

  if permission.can() or model.repository.repository_is_public(namespace, repository):
    tags = model.tag.list_repository_tags(namespace, repository)
    tag_map = {tag.name: tag.image.docker_image_id for tag in tags}
    return jsonify(tag_map)

  abort(403)


@v1_bp.route('/repositories/<path:repository>/tags/<tag>', methods=['GET'])
@process_auth
@anon_protect
@parse_repository_name
def get_tag(namespace, repository, tag):
  permission = ReadRepositoryPermission(namespace, repository)

  if permission.can() or model.repository.repository_is_public(namespace, repository):
    tag_image = model.tag.get_tag_image(namespace, repository, tag)
    resp = make_response('"%s"' % tag_image.docker_image_id)
    resp.headers['Content-Type'] = 'application/json'
    return resp

  abort(403)


@v1_bp.route('/repositories/<path:repository>/tags/<tag>', methods=['PUT'])
@process_auth
@anon_protect
@parse_repository_name
def put_tag(namespace, repository, tag):
  permission = ModifyRepositoryPermission(namespace, repository)

  if permission.can():
    if not TAG_REGEX.match(tag):
      abort(400, TAG_ERROR)

    docker_image_id = json.loads(request.data)
    model.tag.create_or_update_tag(namespace, repository, tag, docker_image_id)

    # Store the updated tag.
    if not 'pushed_tags' in session:
      session['pushed_tags'] = {}

    session['pushed_tags'][tag] = docker_image_id

    return make_response('Created', 200)

  abort(403)


@v1_bp.route('/repositories/<path:repository>/tags/<tag>', methods=['DELETE'])
@process_auth
@anon_protect
@parse_repository_name
def delete_tag(namespace, repository, tag):
  permission = ModifyRepositoryPermission(namespace, repository)

  if permission.can():
    model.tag.delete_tag(namespace, repository, tag)
    return make_response('Deleted', 200)

  abort(403)