/**
 * An element which displays a panel for managing keys for external services.
 */
angular.module('quay').directive('serviceKeysManager', function () {
  var directiveDefinitionObject = {
    priority: 0,
    templateUrl: '/static/directives/service-keys-manager.html',
    replace: false,
    transclude: true,
    restrict: 'C',
    scope: {
      'isEnabled': '=isEnabled'
    },
    controller: function($scope, $element, ApiService, TableService, UIService) {
      $scope.options = {
        'filter': null,
        'predicate': 'expiration_datetime',
        'reverse': false,
      };

      $scope.deleteKeysInfo = null;
      $scope.approveKeysInfo = null;
      $scope.changeKeysInfo = null;

      $scope.checkedKeys = UIService.createCheckStateController([], 'kid');

      $scope.TableService = TableService;
      $scope.newKey = null;
      $scope.creatingKey = false;
      $scope.context = {
        'expirationChangeInfo': null
      };

      var buildOrderedKeys = function() {
        if (!$scope.keys) {
          return;
        }

        var keys = $scope.keys.map(function(key) {
          var expiration_datetime = -Number.MAX_VALUE;
          if (key.rotation_duration) {
            expiration_datetime = -(Number.MAX_VALUE/2);
          } else if (key.expiration_date) {
            expiration_datetime = new Date(key.expiration_date).valueOf() * (-1);
          }

          return $.extend(key, {
            'creation_datetime': new Date(key.creation_date).valueOf() * (-1),
            'expiration_datetime': expiration_datetime,
            'expanded': false
          });
        });

        $scope.orderedKeys = TableService.buildOrderedItems(keys, $scope.options,
            ['name', 'kid', 'service'],
            ['creation_datetime', 'expiration_datetime'])

        $scope.checkedKeys = UIService.createCheckStateController($scope.orderedKeys.visibleEntries, 'kid');
      };

      var loadServiceKeys = function() {
        $scope.options.filter = null;
        $scope.now = new Date();
        $scope.keysResource = ApiService.listServiceKeysAsResource().get(function(resp) {
          $scope.keys = resp['keys'];
          buildOrderedKeys();
        });
      };

      $scope.getKeyTitle = function(key) {
        if (!key) { return ''; }
        return key.name || key.kid.substr(0, 12);
      };

      $scope.toggleDetails = function(key) {
        key.expanded = !key.expanded;
      };

      $scope.getRotationDate = function(key) {
        return moment(key.created_date).add(key.rotation_duration, 's').format('LLL');
      };

      $scope.getExpirationInfo = function(key) {
        if (!key.expiration_date) {
          return '';
        }

        if (key.rotation_duration) {
          var rotate_date = moment(key.created_date).add(key.rotation_duration, 's')
          if (moment().isBefore(rotate_date)) {
            return {'className': 'rotation', 'icon': 'fa-refresh', 'willRotate': true};
          }
        }

        expiration_date = moment(key.expiration_date);
        if (moment().isAfter(expiration_date)) {
          return {'className': 'expired', 'icon': 'fa-warning'};
        }

        if (moment().add(1, 'week').isAfter(expiration_date)) {
          return {'className': 'critical', 'icon': 'fa-warning'};
        }

        if (moment().add(1, 'month').isAfter(expiration_date)) {
          return {'className': 'warning', 'icon': 'fa-warning'};
        }

        return {'className': 'info', 'icon': 'fa-check'};
      };

      $scope.showChangeName = function(key) {
        bootbox.prompt({
          'size': 'small',
          'title': 'Enter a friendly name for key ' + $scope.getKeyTitle(key),
          'value': key.name || '',
          'callback': function(value) {
            if (value != null) {
              var data = {
                'name': value
              };

              var params = {
                'kid': key.kid
              };

              ApiService.updateServiceKey(data, params).then(function(resp) {
                loadServiceKeys();
              }, ApiService.errorDisplay('Could not update service key'));
            }
          }
        });
      };

      $scope.showChangeExpiration = function(key) {
        $scope.context.expirationChangeInfo = {
          'key': key,
          'expiration_date': key.expiration_date ? (new Date(key.expiration_date).getTime() / 1000) : null
        };
      };

      $scope.changeKeyExpiration = function(changeInfo, callback) {
        var errorHandler = ApiService.errorDisplay('Could not change expiration on service key', function() {
          loadServiceKeys();
          callback(false);
        });

        var data = {
          'expiration': changeInfo.expiration_date
        };

        var params = {
          'kid': changeInfo.key.kid
        };

        ApiService.updateServiceKey(data, params).then(function(resp) {
          loadServiceKeys();
          callback(true);
        }, errorHandler);
      };

      $scope.createServiceKey = function() {
        $scope.creatingKey = true;
        ApiService.createServiceKey($scope.newKey).then(function(resp) {
          $scope.creatingKey = false;
          $('#createKeyModal').modal('hide');
          $scope.createdKey = resp;
          $('#createdKeyModal').modal('show');
          loadServiceKeys();
        }, ApiService.errorDisplay('Could not create service key'));
      };

      $scope.showApproveKey = function(key) {
        $scope.approvalKeyInfo = {
          'key': key,
          'notes': ''
        };
      };

      $scope.approveKey = function(approvalKeyInfo, callback) {
        var errorHandler = ApiService.errorDisplay('Could not approve service key', function() {
          loadServiceKeys();
          callback(false);
        });

        var data = {
          'notes': approvalKeyInfo.notes
        };

        var params = {
          'kid': approvalKeyInfo.key.kid
        };

        ApiService.approveServiceKey(data, params).then(function(resp) {
          loadServiceKeys();
          callback(true);
        }, errorHandler);
      };

      $scope.showCreateKey = function() {
        $scope.newKey = {
          'expiration': null
        };

        $('#createKeyModal').modal('show');
      };

      $scope.showDeleteKey = function(key) {
        $scope.deleteKeyInfo = {
          'key': key
        };
      };

      $scope.deleteKey = function(deleteKeyInfo, callback) {
        var errorHandler = ApiService.errorDisplay('Could not delete service key', function() {
          loadServiceKeys();
          callback(false);
        });

        var params = {
          'kid': deleteKeyInfo.key.kid
        };

        ApiService.deleteServiceKey(null, params).then(function(resp) {
          loadServiceKeys();
          callback(true);
        }, errorHandler);
      };

      $scope.isDownloadSupported = function() {
        var isSafari = /^((?!chrome).)*safari/i.test(navigator.userAgent);
        if (isSafari) {
          // Doesn't work properly in Safari, sadly.
          return false;
        }

        try { return !!new Blob(); } catch(e) {}
        return false;
      };

      $scope.downloadPrivateKey = function(key) {
        var blob = new Blob([key.private_key]);
        saveAs(blob, $scope.getKeyTitle(key) + '.pem');
      };

      $scope.askDeleteMultipleKeys = function(keys) {
        $scope.deleteKeysInfo = {
          'keys': keys
        };
      };

      $scope.askApproveMultipleKeys = function(keys) {
        $scope.approveKeysInfo = {
          'keys': keys
        };
      };

      $scope.askChangeExpirationMultipleKeys = function(keys) {
        $scope.changeKeysInfo = {
          'keys': keys
        };
      };

      $scope.allKeyFilter = function(key) {
        return true;
      };

      $scope.noKeyFilter = function(key) {
        return false;
      };

      $scope.unapprovedKeyFilter = function(key) {
        return !key.approval;
      };

      $scope.expiredKeyFilter = function(key) {
        return $scope.getExpirationInfo(key)['className'] == 'expired';
      };

      $scope.allRequireApproval = function(keys) {
        for (var i = 0; i < keys.length; ++i) {
          if (keys[i].approval) {
            return false;
          }
        }

        return true;
      };

      $scope.allExpired = function(keys) {
        for (var i = 0; i < keys.length; ++i) {
          if (!$scope.expiredKeyFilter(keys[i])) {
            return false;
          }
        }

        return true;
      };

      var forAllKeys = function(keys, error_msg, performer, callback) {
        var counter = 0;
        var performAction = function() {
          if (counter >= keys.length) {
            loadServiceKeys();
            callback(true);
            return;
          }

          var key = keys[counter];
          var errorHandler = function(resp) {
            if (resp.status != 404) {
              bootbox.alert(error_msg);
              loadServiceKeys();
              callback(false);
              return;
            }

            performAction();
          };

          counter++;
          performer(key).then(performAction, errorHandler);
        };

        performAction();
      };

      $scope.deleteKeys = function(info, callback) {
        var performer = function(key) {
          var params = {
            'kid': key.kid
          };

          return ApiService.deleteServiceKey(null, params);
        };

        forAllKeys(info.keys, 'Could not delete service key', performer, callback);
      };

      $scope.approveKeys = function(info, callback) {
        var performer = function(key) {
          var params = {
            'kid': key.kid
          };

          var data = {
            'notes': $scope.approveKeysInfo.notes
          };

          return ApiService.approveServiceKey(data, params);
        };

        forAllKeys(info.keys, 'Could not approve service key', performer, callback);
      };

      $scope.changeKeysExpiration = function(info, callback) {
        var performer = function(key) {
          var data = {
            'expiration': info.expiration_date || null
          };

          var params = {
            'kid': key.kid
          };

          return ApiService.updateServiceKey(data, params);
        };

        forAllKeys(info.keys, 'Could not update service key', performer, callback);
      };

      $scope.$watch('options.filter',  buildOrderedKeys);
      $scope.$watch('options.predicate',  buildOrderedKeys);
      $scope.$watch('options.reverse',  buildOrderedKeys);

      $scope.$watch('isEnabled', function(value) {
        if (value) {
          loadServiceKeys();
        }
      });
    }
  };
  return directiveDefinitionObject;
});