diff --git a/flink-runtime-web/web-dashboard/app/scripts/modules/jobs/metrics.svc.coffee b/flink-runtime-web/web-dashboard/app/scripts/modules/jobs/metrics.svc.coffee index e169d6dc519c33..20c743953f2559 100644 --- a/flink-runtime-web/web-dashboard/app/scripts/modules/jobs/metrics.svc.coffee +++ b/flink-runtime-web/web-dashboard/app/scripts/modules/jobs/metrics.svc.coffee @@ -29,17 +29,16 @@ angular.module('flinkApp') } @refresh = $interval => - angular.forEach @watched, (v, jobid) => - angular.forEach v, (nodeid, nk) => - @getAllAvailableMetrics(jobid, nodeid).then (data) => - names = [] - angular.forEach data, (metric, mk) => - names.push metric.id - - if names.length > 0 - @getMetrics(jobid, nodeid, names).then (values) => - if jobid == @observer.jobid && nodeid == @observer.nodeid - @observer.callback(values) if @observer.callback + angular.forEach @metrics, (vertices, jobid) => + angular.forEach vertices, (metrics, nodeid) => + names = [] + angular.forEach metrics, (metric, index) => + names.push metric.id + + if names.length > 0 + @getMetrics(jobid, nodeid, names).then (values) => + if jobid == @observer.jobid && nodeid == @observer.nodeid + @observer.callback(values) if @observer.callback , flinkConfig["refresh-interval"] diff --git a/flink-runtime-web/web-dashboard/web/js/index.js b/flink-runtime-web/web-dashboard/web/js/index.js index 47f9b0c76dcf03..0793f31c417e73 100644 --- a/flink-runtime-web/web-dashboard/web/js/index.js +++ b/flink-runtime-web/web-dashboard/web/js/index.js @@ -1582,24 +1582,22 @@ angular.module('flinkApp').service('MetricsService', ["$http", "$q", "flinkConfi }; this.refresh = $interval((function(_this) { return function() { - return angular.forEach(_this.watched, function(v, jobid) { - return angular.forEach(v, function(nodeid, nk) { - return _this.getAllAvailableMetrics(jobid, nodeid).then(function(data) { - var names; - names = []; - angular.forEach(data, function(metric, mk) { - return names.push(metric.id); - }); - if (names.length > 0) { - return _this.getMetrics(jobid, nodeid, names).then(function(values) { - if (jobid === _this.observer.jobid && nodeid === _this.observer.nodeid) { - if (_this.observer.callback) { - return _this.observer.callback(values); - } - } - }); - } + return angular.forEach(_this.metrics, function(vertices, jobid) { + return angular.forEach(vertices, function(metrics, nodeid) { + var names; + names = []; + angular.forEach(metrics, function(metric, index) { + return names.push(metric.id); }); + if (names.length > 0) { + return _this.getMetrics(jobid, nodeid, names).then(function(values) { + if (jobid === _this.observer.jobid && nodeid === _this.observer.nodeid) { + if (_this.observer.callback) { + return _this.observer.callback(values); + } + } + }); + } }); }); }; @@ -2170,4 +2168,4 @@ angular.module('flinkApp').service('TaskManagersService', ["$http", "flinkConfig return this; }]); -//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["index.coffee","index.js","common/directives.coffee","common/directives.js","common/filters.coffee","common/filters.js","common/services.coffee","common/services.js","modules/jobmanager/jobmanager.ctrl.coffee","modules/jobmanager/jobmanager.ctrl.js","modules/jobmanager/jobmanager.svc.coffee","modules/jobmanager/jobmanager.svc.js","modules/jobs/jobs.ctrl.coffee","modules/jobs/jobs.ctrl.js","modules/jobs/jobs.dir.coffee","modules/jobs/jobs.dir.js","modules/jobs/jobs.svc.coffee","modules/jobs/jobs.svc.js","modules/jobs/metrics.dir.coffee","modules/jobs/metrics.dir.js","modules/jobs/metrics.svc.coffee","modules/jobs/metrics.svc.js","modules/overview/overview.ctrl.coffee","modules/overview/overview.ctrl.js","modules/overview/overview.svc.coffee","modules/overview/overview.svc.js","modules/submit/submit.ctrl.coffee","modules/submit/submit.ctrl.js","modules/submit/submit.svc.coffee","modules/submit/submit.svc.js","modules/taskmanager/taskmanager.ctrl.coffee","modules/taskmanager/taskmanager.ctrl.js","modules/taskmanager/taskmanager.svc.coffee","modules/taskmanager/taskmanager.svc.js"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp', ['ui.router', 'angularMoment', 'dndLists'])\n\n# --------------------------------------\n\n.run ($rootScope) ->\n  $rootScope.sidebarVisible = false\n  $rootScope.showSidebar = ->\n    $rootScope.sidebarVisible = !$rootScope.sidebarVisible\n    $rootScope.sidebarClass = 'force-show'\n\n# --------------------------------------\n\n.value 'flinkConfig', {\n  jobServer: ''\n#  jobServer: 'http://localhost:8081/'\n  \"refresh-interval\": 10000\n}\n\n# --------------------------------------\n\n.run (JobsService, MainService, flinkConfig, $interval) ->\n  MainService.loadConfig().then (config) ->\n    angular.extend flinkConfig, config\n\n    JobsService.listJobs()\n\n    $interval ->\n      JobsService.listJobs()\n    , flinkConfig[\"refresh-interval\"]\n\n\n# --------------------------------------\n\n.config ($uiViewScrollProvider) ->\n  $uiViewScrollProvider.useAnchorScroll()\n\n# --------------------------------------\n\n.run ($rootScope, $state) ->\n  $rootScope.$on '$stateChangeStart', (event, toState, toParams, fromState) ->\n    if toState.redirectTo\n      event.preventDefault()\n      $state.go toState.redirectTo, toParams\n\n# --------------------------------------\n\n.config ($stateProvider, $urlRouterProvider) ->\n  $stateProvider.state \"overview\",\n    url: \"/overview\"\n    views:\n      main:\n        templateUrl: \"partials/overview.html\"\n        controller: 'OverviewController'\n\n  .state \"running-jobs\",\n    url: \"/running-jobs\"\n    views:\n      main:\n        templateUrl: \"partials/jobs/running-jobs.html\"\n        controller: 'RunningJobsController'\n\n  .state \"completed-jobs\",\n    url: \"/completed-jobs\"\n    views:\n      main:\n        templateUrl: \"partials/jobs/completed-jobs.html\"\n        controller: 'CompletedJobsController'\n\n  .state \"single-job\",\n    url: \"/jobs/{jobid}\"\n    abstract: true\n    views:\n      main:\n        templateUrl: \"partials/jobs/job.html\"\n        controller: 'SingleJobController'\n\n  .state \"single-job.plan\",\n    url: \"\"\n    redirectTo: \"single-job.plan.subtasks\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.plan.html\"\n        controller: 'JobPlanController'\n\n  .state \"single-job.plan.subtasks\",\n    url: \"\"\n    views:\n      'node-details':\n        templateUrl: \"partials/jobs/job.plan.node-list.subtasks.html\"\n        controller: 'JobPlanSubtasksController'\n\n  .state \"single-job.plan.metrics\",\n    url: \"/metrics\"\n    views:\n      'node-details':\n        templateUrl: \"partials/jobs/job.plan.node-list.metrics.html\"\n        controller: 'JobPlanMetricsController'\n\n  .state \"single-job.plan.taskmanagers\",\n    url: \"/taskmanagers\"\n    views:\n      'node-details':\n        templateUrl: \"partials/jobs/job.plan.node-list.taskmanagers.html\"\n        controller: 'JobPlanTaskManagersController'\n\n  .state \"single-job.plan.accumulators\",\n    url: \"/accumulators\"\n    views:\n      'node-details':\n        templateUrl: \"partials/jobs/job.plan.node-list.accumulators.html\"\n        controller: 'JobPlanAccumulatorsController'\n\n  .state \"single-job.plan.checkpoints\",\n    url: \"/checkpoints\"\n    views:\n      'node-details':\n        templateUrl: \"partials/jobs/job.plan.node-list.checkpoints.html\"\n        controller: 'JobPlanCheckpointsController'\n\n  .state \"single-job.plan.backpressure\",\n    url: \"/backpressure\"\n    views:\n      'node-details':\n        templateUrl: \"partials/jobs/job.plan.node-list.backpressure.html\"\n        controller: 'JobPlanBackPressureController'\n\n  .state \"single-job.timeline\",\n    url: \"/timeline\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.timeline.html\"\n\n  .state \"single-job.timeline.vertex\",\n    url: \"/{vertexId}\"\n    views:\n      vertex:\n        templateUrl: \"partials/jobs/job.timeline.vertex.html\"\n        controller: 'JobTimelineVertexController'\n\n  .state \"single-job.exceptions\",\n    url: \"/exceptions\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.exceptions.html\"\n        controller: 'JobExceptionsController'\n\n  .state \"single-job.config\",\n    url: \"/config\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.config.html\"\n\n  .state \"all-manager\",\n    url: \"/taskmanagers\"\n    views:\n      main:\n        templateUrl: \"partials/taskmanager/index.html\"\n        controller: 'AllTaskManagersController'\n\n  .state \"single-manager\",\n      url: \"/taskmanager/{taskmanagerid}\"\n      views:\n        main:\n          templateUrl: \"partials/taskmanager/taskmanager.html\"\n\n  .state \"single-manager.metrics\",\n    url: \"/metrics\"\n    views:\n      details:\n        templateUrl: \"partials/taskmanager/taskmanager.metrics.html\"\n        controller: 'SingleTaskManagerController'\n\n  .state \"single-manager.stdout\",\n    url: \"/stdout\"\n    views:\n      details:\n        templateUrl: \"partials/taskmanager/taskmanager.stdout.html\"\n        controller: 'SingleTaskManagerStdoutController'\n\n  .state \"single-manager.log\",\n    url: \"/log\"\n    views:\n      details:\n        templateUrl: \"partials/taskmanager/taskmanager.log.html\"\n        controller: 'SingleTaskManagerLogsController'\n\n  .state \"jobmanager\",\n      url: \"/jobmanager\"\n      views:\n        main:\n          templateUrl: \"partials/jobmanager/index.html\"\n\n  .state \"jobmanager.config\",\n    url: \"/config\"\n    views:\n      details:\n        templateUrl: \"partials/jobmanager/config.html\"\n        controller: 'JobManagerConfigController'\n\n  .state \"jobmanager.stdout\",\n    url: \"/stdout\"\n    views:\n      details:\n        templateUrl: \"partials/jobmanager/stdout.html\"\n        controller: 'JobManagerStdoutController'\n\n  .state \"jobmanager.log\",\n    url: \"/log\"\n    views:\n      details:\n        templateUrl: \"partials/jobmanager/log.html\"\n        controller: 'JobManagerLogsController'\n\n  .state \"submit\",\n      url: \"/submit\"\n      views:\n        main:\n          templateUrl: \"partials/submit.html\"\n          controller: \"JobSubmitController\"\n\n  $urlRouterProvider.otherwise \"/overview\"\n","angular.module('flinkApp', ['ui.router', 'angularMoment', 'dndLists']).run(function($rootScope) {\n  $rootScope.sidebarVisible = false;\n  return $rootScope.showSidebar = function() {\n    $rootScope.sidebarVisible = !$rootScope.sidebarVisible;\n    return $rootScope.sidebarClass = 'force-show';\n  };\n}).value('flinkConfig', {\n  jobServer: '',\n  \"refresh-interval\": 10000\n}).run(function(JobsService, MainService, flinkConfig, $interval) {\n  return MainService.loadConfig().then(function(config) {\n    angular.extend(flinkConfig, config);\n    JobsService.listJobs();\n    return $interval(function() {\n      return JobsService.listJobs();\n    }, flinkConfig[\"refresh-interval\"]);\n  });\n}).config(function($uiViewScrollProvider) {\n  return $uiViewScrollProvider.useAnchorScroll();\n}).run(function($rootScope, $state) {\n  return $rootScope.$on('$stateChangeStart', function(event, toState, toParams, fromState) {\n    if (toState.redirectTo) {\n      event.preventDefault();\n      return $state.go(toState.redirectTo, toParams);\n    }\n  });\n}).config(function($stateProvider, $urlRouterProvider) {\n  $stateProvider.state(\"overview\", {\n    url: \"/overview\",\n    views: {\n      main: {\n        templateUrl: \"partials/overview.html\",\n        controller: 'OverviewController'\n      }\n    }\n  }).state(\"running-jobs\", {\n    url: \"/running-jobs\",\n    views: {\n      main: {\n        templateUrl: \"partials/jobs/running-jobs.html\",\n        controller: 'RunningJobsController'\n      }\n    }\n  }).state(\"completed-jobs\", {\n    url: \"/completed-jobs\",\n    views: {\n      main: {\n        templateUrl: \"partials/jobs/completed-jobs.html\",\n        controller: 'CompletedJobsController'\n      }\n    }\n  }).state(\"single-job\", {\n    url: \"/jobs/{jobid}\",\n    abstract: true,\n    views: {\n      main: {\n        templateUrl: \"partials/jobs/job.html\",\n        controller: 'SingleJobController'\n      }\n    }\n  }).state(\"single-job.plan\", {\n    url: \"\",\n    redirectTo: \"single-job.plan.subtasks\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.plan.html\",\n        controller: 'JobPlanController'\n      }\n    }\n  }).state(\"single-job.plan.subtasks\", {\n    url: \"\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.subtasks.html\",\n        controller: 'JobPlanSubtasksController'\n      }\n    }\n  }).state(\"single-job.plan.metrics\", {\n    url: \"/metrics\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.metrics.html\",\n        controller: 'JobPlanMetricsController'\n      }\n    }\n  }).state(\"single-job.plan.taskmanagers\", {\n    url: \"/taskmanagers\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.taskmanagers.html\",\n        controller: 'JobPlanTaskManagersController'\n      }\n    }\n  }).state(\"single-job.plan.accumulators\", {\n    url: \"/accumulators\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.accumulators.html\",\n        controller: 'JobPlanAccumulatorsController'\n      }\n    }\n  }).state(\"single-job.plan.checkpoints\", {\n    url: \"/checkpoints\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.checkpoints.html\",\n        controller: 'JobPlanCheckpointsController'\n      }\n    }\n  }).state(\"single-job.plan.backpressure\", {\n    url: \"/backpressure\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.backpressure.html\",\n        controller: 'JobPlanBackPressureController'\n      }\n    }\n  }).state(\"single-job.timeline\", {\n    url: \"/timeline\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.timeline.html\"\n      }\n    }\n  }).state(\"single-job.timeline.vertex\", {\n    url: \"/{vertexId}\",\n    views: {\n      vertex: {\n        templateUrl: \"partials/jobs/job.timeline.vertex.html\",\n        controller: 'JobTimelineVertexController'\n      }\n    }\n  }).state(\"single-job.exceptions\", {\n    url: \"/exceptions\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.exceptions.html\",\n        controller: 'JobExceptionsController'\n      }\n    }\n  }).state(\"single-job.config\", {\n    url: \"/config\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.config.html\"\n      }\n    }\n  }).state(\"all-manager\", {\n    url: \"/taskmanagers\",\n    views: {\n      main: {\n        templateUrl: \"partials/taskmanager/index.html\",\n        controller: 'AllTaskManagersController'\n      }\n    }\n  }).state(\"single-manager\", {\n    url: \"/taskmanager/{taskmanagerid}\",\n    views: {\n      main: {\n        templateUrl: \"partials/taskmanager/taskmanager.html\"\n      }\n    }\n  }).state(\"single-manager.metrics\", {\n    url: \"/metrics\",\n    views: {\n      details: {\n        templateUrl: \"partials/taskmanager/taskmanager.metrics.html\",\n        controller: 'SingleTaskManagerController'\n      }\n    }\n  }).state(\"single-manager.stdout\", {\n    url: \"/stdout\",\n    views: {\n      details: {\n        templateUrl: \"partials/taskmanager/taskmanager.stdout.html\",\n        controller: 'SingleTaskManagerStdoutController'\n      }\n    }\n  }).state(\"single-manager.log\", {\n    url: \"/log\",\n    views: {\n      details: {\n        templateUrl: \"partials/taskmanager/taskmanager.log.html\",\n        controller: 'SingleTaskManagerLogsController'\n      }\n    }\n  }).state(\"jobmanager\", {\n    url: \"/jobmanager\",\n    views: {\n      main: {\n        templateUrl: \"partials/jobmanager/index.html\"\n      }\n    }\n  }).state(\"jobmanager.config\", {\n    url: \"/config\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobmanager/config.html\",\n        controller: 'JobManagerConfigController'\n      }\n    }\n  }).state(\"jobmanager.stdout\", {\n    url: \"/stdout\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobmanager/stdout.html\",\n        controller: 'JobManagerStdoutController'\n      }\n    }\n  }).state(\"jobmanager.log\", {\n    url: \"/log\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobmanager/log.html\",\n        controller: 'JobManagerLogsController'\n      }\n    }\n  }).state(\"submit\", {\n    url: \"/submit\",\n    views: {\n      main: {\n        templateUrl: \"partials/submit.html\",\n        controller: \"JobSubmitController\"\n      }\n    }\n  });\n  return $urlRouterProvider.otherwise(\"/overview\");\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n# ----------------------------------------------\n\n.directive 'bsLabel', (JobsService) ->\n  transclude: true\n  replace: true\n  scope: \n    getLabelClass: \"&\"\n    status: \"@\"\n\n  template: \"<span title='{{status}}' ng-class='getLabelClass()'><ng-transclude></ng-transclude></span>\"\n  \n  link: (scope, element, attrs) ->\n    scope.getLabelClass = ->\n      'label label-' + JobsService.translateLabelState(attrs.status)\n\n# ----------------------------------------------\n\n.directive 'bpLabel', (JobsService) ->\n  transclude: true\n  replace: true\n  scope:\n    getBackPressureLabelClass: \"&\"\n    status: \"@\"\n\n  template: \"<span title='{{status}}' ng-class='getBackPressureLabelClass()'><ng-transclude></ng-transclude></span>\"\n\n  link: (scope, element, attrs) ->\n    scope.getBackPressureLabelClass = ->\n      'label label-' + JobsService.translateBackPressureLabelState(attrs.status)\n\n# ----------------------------------------------\n\n.directive 'indicatorPrimary', (JobsService) ->\n  replace: true\n  scope: \n    getLabelClass: \"&\"\n    status: '@'\n\n  template: \"<i title='{{status}}' ng-class='getLabelClass()' />\"\n  \n  link: (scope, element, attrs) ->\n    scope.getLabelClass = ->\n      'fa fa-circle indicator indicator-' + JobsService.translateLabelState(attrs.status)\n\n# ----------------------------------------------\n\n.directive 'tableProperty', ->\n  replace: true\n  scope:\n    value: '='\n\n  template: \"<td title=\\\"{{value || 'None'}}\\\">{{value || 'None'}}</td>\"\n","angular.module('flinkApp').directive('bsLabel', function(JobsService) {\n  return {\n    transclude: true,\n    replace: true,\n    scope: {\n      getLabelClass: \"&\",\n      status: \"@\"\n    },\n    template: \"<span title='{{status}}' ng-class='getLabelClass()'><ng-transclude></ng-transclude></span>\",\n    link: function(scope, element, attrs) {\n      return scope.getLabelClass = function() {\n        return 'label label-' + JobsService.translateLabelState(attrs.status);\n      };\n    }\n  };\n}).directive('bpLabel', function(JobsService) {\n  return {\n    transclude: true,\n    replace: true,\n    scope: {\n      getBackPressureLabelClass: \"&\",\n      status: \"@\"\n    },\n    template: \"<span title='{{status}}' ng-class='getBackPressureLabelClass()'><ng-transclude></ng-transclude></span>\",\n    link: function(scope, element, attrs) {\n      return scope.getBackPressureLabelClass = function() {\n        return 'label label-' + JobsService.translateBackPressureLabelState(attrs.status);\n      };\n    }\n  };\n}).directive('indicatorPrimary', function(JobsService) {\n  return {\n    replace: true,\n    scope: {\n      getLabelClass: \"&\",\n      status: '@'\n    },\n    template: \"<i title='{{status}}' ng-class='getLabelClass()' />\",\n    link: function(scope, element, attrs) {\n      return scope.getLabelClass = function() {\n        return 'fa fa-circle indicator indicator-' + JobsService.translateLabelState(attrs.status);\n      };\n    }\n  };\n}).directive('tableProperty', function() {\n  return {\n    replace: true,\n    scope: {\n      value: '='\n    },\n    template: \"<td title=\\\"{{value || 'None'}}\\\">{{value || 'None'}}</td>\"\n  };\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.filter \"amDurationFormatExtended\", (angularMomentConfig) ->\n  amDurationFormatExtendedFilter = (value, format, durationFormat) ->\n    return \"\"  if typeof value is \"undefined\" or value is null\n\n    moment.duration(value, format).format(durationFormat, { trim: false })\n\n  amDurationFormatExtendedFilter.$stateful = angularMomentConfig.statefulFilters\n\n  amDurationFormatExtendedFilter\n\n.filter \"humanizeDuration\", ->\n  (value, short) ->\n    return \"\" if typeof value is \"undefined\" or value is null\n    ms = value % 1000\n    x = Math.floor(value / 1000)\n    seconds = x % 60\n    x = Math.floor(x / 60)\n    minutes = x % 60\n    x = Math.floor(x / 60)\n    hours = x % 24\n    x = Math.floor(x / 24)\n    days = x\n    if days == 0\n      if hours == 0\n        if minutes == 0\n          if seconds == 0\n            return ms + \"ms\"\n          else\n            return seconds + \"s \"\n        else\n          return minutes + \"m \" + seconds + \"s\"\n      else\n        if short then return hours + \"h \" + minutes + \"m\" else return hours + \"h \" + minutes + \"m \" + seconds + \"s\"\n    else\n      if short then return days + \"d \" + hours + \"h\" else return days + \"d \" + hours + \"h \" + minutes + \"m \" + seconds + \"s\"\n\n.filter \"humanizeText\", ->\n  (text) ->\n    # TODO: extend... a lot\n    if text then text.replace(/&gt;/g, \">\").replace(/<br\\/>/g,\"\") else ''\n\n.filter \"humanizeBytes\", ->\n  (bytes) ->\n    units = [\"B\", \"KB\", \"MB\", \"GB\", \"TB\", \"PB\", \"EB\"]\n    converter = (value, power) ->\n      base = Math.pow(1024, power)\n      if value < base\n        return (value / base).toFixed(2) + \" \" + units[power]\n      else if value < base * 1000\n        return (value / base).toPrecision(3) + \" \" + units[power]\n      else\n        return converter(value, power + 1)\n    return \"\" if typeof bytes is \"undefined\" or bytes is null\n    if bytes < 1000 then bytes + \" B\" else converter(bytes, 1)\n\n.filter \"toLocaleString\", ->\n  (text) -> text.toLocaleString()\n\n.filter \"toUpperCase\", ->\n  (text) -> text.toUpperCase()\n","angular.module('flinkApp').filter(\"amDurationFormatExtended\", function(angularMomentConfig) {\n  var amDurationFormatExtendedFilter;\n  amDurationFormatExtendedFilter = function(value, format, durationFormat) {\n    if (typeof value === \"undefined\" || value === null) {\n      return \"\";\n    }\n    return moment.duration(value, format).format(durationFormat, {\n      trim: false\n    });\n  };\n  amDurationFormatExtendedFilter.$stateful = angularMomentConfig.statefulFilters;\n  return amDurationFormatExtendedFilter;\n}).filter(\"humanizeDuration\", function() {\n  return function(value, short) {\n    var days, hours, minutes, ms, seconds, x;\n    if (typeof value === \"undefined\" || value === null) {\n      return \"\";\n    }\n    ms = value % 1000;\n    x = Math.floor(value / 1000);\n    seconds = x % 60;\n    x = Math.floor(x / 60);\n    minutes = x % 60;\n    x = Math.floor(x / 60);\n    hours = x % 24;\n    x = Math.floor(x / 24);\n    days = x;\n    if (days === 0) {\n      if (hours === 0) {\n        if (minutes === 0) {\n          if (seconds === 0) {\n            return ms + \"ms\";\n          } else {\n            return seconds + \"s \";\n          }\n        } else {\n          return minutes + \"m \" + seconds + \"s\";\n        }\n      } else {\n        if (short) {\n          return hours + \"h \" + minutes + \"m\";\n        } else {\n          return hours + \"h \" + minutes + \"m \" + seconds + \"s\";\n        }\n      }\n    } else {\n      if (short) {\n        return days + \"d \" + hours + \"h\";\n      } else {\n        return days + \"d \" + hours + \"h \" + minutes + \"m \" + seconds + \"s\";\n      }\n    }\n  };\n}).filter(\"humanizeText\", function() {\n  return function(text) {\n    if (text) {\n      return text.replace(/&gt;/g, \">\").replace(/<br\\/>/g, \"\");\n    } else {\n      return '';\n    }\n  };\n}).filter(\"humanizeBytes\", function() {\n  return function(bytes) {\n    var converter, units;\n    units = [\"B\", \"KB\", \"MB\", \"GB\", \"TB\", \"PB\", \"EB\"];\n    converter = function(value, power) {\n      var base;\n      base = Math.pow(1024, power);\n      if (value < base) {\n        return (value / base).toFixed(2) + \" \" + units[power];\n      } else if (value < base * 1000) {\n        return (value / base).toPrecision(3) + \" \" + units[power];\n      } else {\n        return converter(value, power + 1);\n      }\n    };\n    if (typeof bytes === \"undefined\" || bytes === null) {\n      return \"\";\n    }\n    if (bytes < 1000) {\n      return bytes + \" B\";\n    } else {\n      return converter(bytes, 1);\n    }\n  };\n}).filter(\"toLocaleString\", function() {\n  return function(text) {\n    return text.toLocaleString();\n  };\n}).filter(\"toUpperCase\", function() {\n  return function(text) {\n    return text.toUpperCase();\n  };\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.service 'MainService', ($http, flinkConfig, $q) ->\n  @loadConfig = ->\n    deferred = $q.defer()\n\n    $http.get flinkConfig.jobServer + \"config\"\n    .success (data, status, headers, config) ->\n      deferred.resolve(data)\n\n    deferred.promise\n\n\n  @\n","angular.module('flinkApp').service('MainService', function($http, flinkConfig, $q) {\n  this.loadConfig = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"config\").success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.controller 'JobManagerConfigController', ($scope, JobManagerConfigService) ->\n  JobManagerConfigService.loadConfig().then (data) ->\n    if !$scope.jobmanager?\n      $scope.jobmanager = {}\n    $scope.jobmanager['config'] = data\n\n.controller 'JobManagerLogsController', ($scope, JobManagerLogsService) ->\n  JobManagerLogsService.loadLogs().then (data) ->\n    if !$scope.jobmanager?\n      $scope.jobmanager = {}\n    $scope.jobmanager['log'] = data\n\n  $scope.reloadData = () ->\n    JobManagerLogsService.loadLogs().then (data) ->\n      $scope.jobmanager['log'] = data\n\n.controller 'JobManagerStdoutController', ($scope, JobManagerStdoutService) ->\n  JobManagerStdoutService.loadStdout().then (data) ->\n    if !$scope.jobmanager?\n      $scope.jobmanager = {}\n    $scope.jobmanager['stdout'] = data\n\n  $scope.reloadData = () ->\n    JobManagerStdoutService.loadStdout().then (data) ->\n      $scope.jobmanager['stdout'] = data\n","angular.module('flinkApp').controller('JobManagerConfigController', function($scope, JobManagerConfigService) {\n  return JobManagerConfigService.loadConfig().then(function(data) {\n    if ($scope.jobmanager == null) {\n      $scope.jobmanager = {};\n    }\n    return $scope.jobmanager['config'] = data;\n  });\n}).controller('JobManagerLogsController', function($scope, JobManagerLogsService) {\n  JobManagerLogsService.loadLogs().then(function(data) {\n    if ($scope.jobmanager == null) {\n      $scope.jobmanager = {};\n    }\n    return $scope.jobmanager['log'] = data;\n  });\n  return $scope.reloadData = function() {\n    return JobManagerLogsService.loadLogs().then(function(data) {\n      return $scope.jobmanager['log'] = data;\n    });\n  };\n}).controller('JobManagerStdoutController', function($scope, JobManagerStdoutService) {\n  JobManagerStdoutService.loadStdout().then(function(data) {\n    if ($scope.jobmanager == null) {\n      $scope.jobmanager = {};\n    }\n    return $scope.jobmanager['stdout'] = data;\n  });\n  return $scope.reloadData = function() {\n    return JobManagerStdoutService.loadStdout().then(function(data) {\n      return $scope.jobmanager['stdout'] = data;\n    });\n  };\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.service 'JobManagerConfigService', ($http, flinkConfig, $q) ->\n  config = {}\n\n  @loadConfig = ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"jobmanager/config\")\n    .success (data, status, headers, config) ->\n      config = data\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @\n\n.service 'JobManagerLogsService', ($http, flinkConfig, $q) ->\n  logs = {}\n\n  @loadLogs = ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"jobmanager/log\")\n    .success (data, status, headers, config) ->\n      logs = data\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @\n\n.service 'JobManagerStdoutService', ($http, flinkConfig, $q) ->\n  stdout = {}\n\n  @loadStdout = ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"jobmanager/stdout\")\n    .success (data, status, headers, config) ->\n      stdout = data\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @\n","angular.module('flinkApp').service('JobManagerConfigService', function($http, flinkConfig, $q) {\n  var config;\n  config = {};\n  this.loadConfig = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobmanager/config\").success(function(data, status, headers, config) {\n      config = data;\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n}).service('JobManagerLogsService', function($http, flinkConfig, $q) {\n  var logs;\n  logs = {};\n  this.loadLogs = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobmanager/log\").success(function(data, status, headers, config) {\n      logs = data;\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n}).service('JobManagerStdoutService', function($http, flinkConfig, $q) {\n  var stdout;\n  stdout = {};\n  this.loadStdout = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobmanager/stdout\").success(function(data, status, headers, config) {\n      stdout = data;\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.controller 'RunningJobsController', ($scope, $state, $stateParams, JobsService) ->\n  $scope.jobObserver = ->\n    $scope.jobs = JobsService.getJobs('running')\n\n  JobsService.registerObserver($scope.jobObserver)\n  $scope.$on '$destroy', ->\n    JobsService.unRegisterObserver($scope.jobObserver)\n\n  $scope.jobObserver()\n\n# --------------------------------------\n\n.controller 'CompletedJobsController', ($scope, $state, $stateParams, JobsService) ->\n  $scope.jobObserver = ->\n    $scope.jobs = JobsService.getJobs('finished')\n\n  JobsService.registerObserver($scope.jobObserver)\n  $scope.$on '$destroy', ->\n    JobsService.unRegisterObserver($scope.jobObserver)\n\n  $scope.jobObserver()\n\n# --------------------------------------\n\n.controller 'SingleJobController', ($scope, $state, $stateParams, JobsService, MetricsService, $rootScope, flinkConfig, $interval) ->\n  $scope.jobid = $stateParams.jobid\n  $scope.job = null\n  $scope.plan = null\n  $scope.vertices = null\n  $scope.jobCheckpointStats = null\n  $scope.showHistory = false\n  $scope.backPressureOperatorStats = {}\n\n  JobsService.loadJob($stateParams.jobid).then (data) ->\n    $scope.job = data\n    $scope.plan = data.plan\n    $scope.vertices = data.vertices\n    MetricsService.setupMetrics($stateParams.jobid, data.vertices)\n\n  refresher = $interval ->\n    JobsService.loadJob($stateParams.jobid).then (data) ->\n      $scope.job = data\n\n      $scope.$broadcast 'reload'\n\n  , flinkConfig[\"refresh-interval\"]\n\n  $scope.$on '$destroy', ->\n    $scope.job = null\n    $scope.plan = null\n    $scope.vertices = null\n    $scope.jobCheckpointStats = null\n    $scope.backPressureOperatorStats = null\n\n    $interval.cancel(refresher)\n\n  $scope.cancelJob = (cancelEvent) ->\n    angular.element(cancelEvent.currentTarget).removeClass(\"btn\").removeClass(\"btn-default\").html('Cancelling...')\n    JobsService.cancelJob($stateParams.jobid).then (data) ->\n      {}\n\n  $scope.stopJob = (stopEvent) ->\n    angular.element(stopEvent.currentTarget).removeClass(\"btn\").removeClass(\"btn-default\").html('Stopping...')\n    JobsService.stopJob($stateParams.jobid).then (data) ->\n      {}\n\n  $scope.toggleHistory = ->\n    $scope.showHistory = !$scope.showHistory\n\n# --------------------------------------\n\n.controller 'JobPlanController', ($scope, $state, $stateParams, $window, JobsService) ->\n  $scope.nodeid = null\n  $scope.nodeUnfolded = false\n  $scope.stateList = JobsService.stateList()\n\n  $scope.changeNode = (nodeid) ->\n    if nodeid != $scope.nodeid\n      $scope.nodeid = nodeid\n      $scope.vertex = null\n      $scope.subtasks = null\n      $scope.accumulators = null\n      $scope.operatorCheckpointStats = null\n\n      $scope.$broadcast 'reload'\n      $scope.$broadcast 'node:change', $scope.nodeid\n\n    else\n      $scope.nodeid = null\n      $scope.nodeUnfolded = false\n      $scope.vertex = null\n      $scope.subtasks = null\n      $scope.accumulators = null\n      $scope.operatorCheckpointStats = null\n\n  $scope.deactivateNode = ->\n    $scope.nodeid = null\n    $scope.nodeUnfolded = false\n    $scope.vertex = null\n    $scope.subtasks = null\n    $scope.accumulators = null\n    $scope.operatorCheckpointStats = null\n\n  $scope.toggleFold = ->\n    $scope.nodeUnfolded = !$scope.nodeUnfolded\n\n# --------------------------------------\n\n.controller 'JobPlanSubtasksController', ($scope, JobsService) ->\n  getSubtasks = ->\n    JobsService.getSubtasks($scope.nodeid).then (data) ->\n      $scope.subtasks = data\n\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.st)\n    getSubtasks()\n\n  $scope.$on 'reload', (event) ->\n    getSubtasks() if $scope.nodeid\n\n# --------------------------------------\n\n.controller 'JobPlanTaskManagersController', ($scope, JobsService) ->\n  getTaskManagers = ->\n    JobsService.getTaskManagers($scope.nodeid).then (data) ->\n      $scope.taskmanagers = data\n\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.st)\n    getTaskManagers()\n\n  $scope.$on 'reload', (event) ->\n    getTaskManagers() if $scope.nodeid\n\n# --------------------------------------\n\n.controller 'JobPlanAccumulatorsController', ($scope, JobsService) ->\n  getAccumulators = ->\n    JobsService.getAccumulators($scope.nodeid).then (data) ->\n      $scope.accumulators = data.main\n      $scope.subtaskAccumulators = data.subtasks\n\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.accumulators)\n    getAccumulators()\n\n  $scope.$on 'reload', (event) ->\n    getAccumulators() if $scope.nodeid\n\n# --------------------------------------\n\n.controller 'JobPlanCheckpointsController', ($scope, JobsService) ->\n  getJobCheckpointStats = ->\n    JobsService.getJobCheckpointStats($scope.jobid).then (data) ->\n      $scope.jobCheckpointStats = data\n\n  getOperatorCheckpointStats = ->\n    JobsService.getOperatorCheckpointStats($scope.nodeid).then (data) ->\n      $scope.operatorCheckpointStats = data.operatorStats\n      $scope.subtasksCheckpointStats = data.subtasksStats\n\n  # Get the per job stats\n  getJobCheckpointStats()\n\n  # Get the per operator stats\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.operatorCheckpointStats)\n    getOperatorCheckpointStats()\n\n  $scope.$on 'reload', (event) ->\n    getJobCheckpointStats()\n    getOperatorCheckpointStats() if $scope.nodeid\n\n# --------------------------------------\n\n.controller 'JobPlanBackPressureController', ($scope, JobsService) ->\n  getOperatorBackPressure = ->\n    $scope.now = Date.now()\n\n    if $scope.nodeid\n      JobsService.getOperatorBackPressure($scope.nodeid).then (data) ->\n        $scope.backPressureOperatorStats[$scope.nodeid] = data\n\n  getOperatorBackPressure()\n\n  $scope.$on 'reload', (event) ->\n    getOperatorBackPressure()\n\n# --------------------------------------\n\n.controller 'JobTimelineVertexController', ($scope, $state, $stateParams, JobsService) ->\n  getVertex = ->\n    JobsService.getVertex($stateParams.vertexId).then (data) ->\n      $scope.vertex = data\n\n  getVertex()\n\n  $scope.$on 'reload', (event) ->\n    getVertex()\n\n# --------------------------------------\n\n.controller 'JobExceptionsController', ($scope, $state, $stateParams, JobsService) ->\n  JobsService.loadExceptions().then (data) ->\n    $scope.exceptions = data\n\n# --------------------------------------\n\n.controller 'JobPropertiesController', ($scope, JobsService) ->\n  $scope.changeNode = (nodeid) ->\n    if nodeid != $scope.nodeid\n      $scope.nodeid = nodeid\n\n      JobsService.getNode(nodeid).then (data) ->\n        $scope.node = data\n\n    else\n      $scope.nodeid = null\n      $scope.node = null\n\n# --------------------------------------\n\n.controller 'JobPlanMetricsController', ($scope, JobsService, MetricsService) ->\n  $scope.dragging = false\n  $scope.window = MetricsService.getWindow()\n  $scope.availableMetrics = null\n\n  $scope.$on '$destroy', ->\n    MetricsService.unRegisterObserver()\n\n  loadMetrics = ->\n    JobsService.getVertex($scope.nodeid).then (data) ->\n      $scope.vertex = data\n\n    MetricsService.getAvailableMetrics($scope.jobid, $scope.nodeid).then (data) ->\n      $scope.availableMetrics = data\n      $scope.metrics = MetricsService.getMetricsSetup($scope.jobid, $scope.nodeid).names\n\n      MetricsService.registerObserver($scope.jobid, $scope.nodeid, (data) ->\n        $scope.$broadcast \"metrics:data:update\", data.timestamp, data.values\n      )\n\n  $scope.dropped = (event, index, item, external, type) ->\n\n    MetricsService.orderMetrics($scope.jobid, $scope.nodeid, item, index)\n    $scope.$broadcast \"metrics:refresh\", item\n    loadMetrics()\n    false\n\n  $scope.dragStart = ->\n    $scope.dragging = true\n\n  $scope.dragEnd = ->\n    $scope.dragging = false\n\n  $scope.addMetric = (metric) ->\n    MetricsService.addMetric($scope.jobid, $scope.nodeid, metric.id)\n    loadMetrics()\n\n  $scope.removeMetric = (metric) ->\n    MetricsService.removeMetric($scope.jobid, $scope.nodeid, metric)\n    loadMetrics()\n\n  $scope.setMetricSize = (metric, size) ->\n    MetricsService.setMetricSize($scope.jobid, $scope.nodeid, metric, size)\n    loadMetrics()\n\n  $scope.getValues = (metric) ->\n    MetricsService.getValues($scope.jobid, $scope.nodeid, metric)\n\n  $scope.$on 'node:change', (event, nodeid) ->\n    loadMetrics() if !$scope.dragging\n\n  loadMetrics() if $scope.nodeid\n\n# --------------------------------------\n","angular.module('flinkApp').controller('RunningJobsController', function($scope, $state, $stateParams, JobsService) {\n  $scope.jobObserver = function() {\n    return $scope.jobs = JobsService.getJobs('running');\n  };\n  JobsService.registerObserver($scope.jobObserver);\n  $scope.$on('$destroy', function() {\n    return JobsService.unRegisterObserver($scope.jobObserver);\n  });\n  return $scope.jobObserver();\n}).controller('CompletedJobsController', function($scope, $state, $stateParams, JobsService) {\n  $scope.jobObserver = function() {\n    return $scope.jobs = JobsService.getJobs('finished');\n  };\n  JobsService.registerObserver($scope.jobObserver);\n  $scope.$on('$destroy', function() {\n    return JobsService.unRegisterObserver($scope.jobObserver);\n  });\n  return $scope.jobObserver();\n}).controller('SingleJobController', function($scope, $state, $stateParams, JobsService, MetricsService, $rootScope, flinkConfig, $interval) {\n  var refresher;\n  $scope.jobid = $stateParams.jobid;\n  $scope.job = null;\n  $scope.plan = null;\n  $scope.vertices = null;\n  $scope.jobCheckpointStats = null;\n  $scope.showHistory = false;\n  $scope.backPressureOperatorStats = {};\n  JobsService.loadJob($stateParams.jobid).then(function(data) {\n    $scope.job = data;\n    $scope.plan = data.plan;\n    $scope.vertices = data.vertices;\n    return MetricsService.setupMetrics($stateParams.jobid, data.vertices);\n  });\n  refresher = $interval(function() {\n    return JobsService.loadJob($stateParams.jobid).then(function(data) {\n      $scope.job = data;\n      return $scope.$broadcast('reload');\n    });\n  }, flinkConfig[\"refresh-interval\"]);\n  $scope.$on('$destroy', function() {\n    $scope.job = null;\n    $scope.plan = null;\n    $scope.vertices = null;\n    $scope.jobCheckpointStats = null;\n    $scope.backPressureOperatorStats = null;\n    return $interval.cancel(refresher);\n  });\n  $scope.cancelJob = function(cancelEvent) {\n    angular.element(cancelEvent.currentTarget).removeClass(\"btn\").removeClass(\"btn-default\").html('Cancelling...');\n    return JobsService.cancelJob($stateParams.jobid).then(function(data) {\n      return {};\n    });\n  };\n  $scope.stopJob = function(stopEvent) {\n    angular.element(stopEvent.currentTarget).removeClass(\"btn\").removeClass(\"btn-default\").html('Stopping...');\n    return JobsService.stopJob($stateParams.jobid).then(function(data) {\n      return {};\n    });\n  };\n  return $scope.toggleHistory = function() {\n    return $scope.showHistory = !$scope.showHistory;\n  };\n}).controller('JobPlanController', function($scope, $state, $stateParams, $window, JobsService) {\n  $scope.nodeid = null;\n  $scope.nodeUnfolded = false;\n  $scope.stateList = JobsService.stateList();\n  $scope.changeNode = function(nodeid) {\n    if (nodeid !== $scope.nodeid) {\n      $scope.nodeid = nodeid;\n      $scope.vertex = null;\n      $scope.subtasks = null;\n      $scope.accumulators = null;\n      $scope.operatorCheckpointStats = null;\n      $scope.$broadcast('reload');\n      return $scope.$broadcast('node:change', $scope.nodeid);\n    } else {\n      $scope.nodeid = null;\n      $scope.nodeUnfolded = false;\n      $scope.vertex = null;\n      $scope.subtasks = null;\n      $scope.accumulators = null;\n      return $scope.operatorCheckpointStats = null;\n    }\n  };\n  $scope.deactivateNode = function() {\n    $scope.nodeid = null;\n    $scope.nodeUnfolded = false;\n    $scope.vertex = null;\n    $scope.subtasks = null;\n    $scope.accumulators = null;\n    return $scope.operatorCheckpointStats = null;\n  };\n  return $scope.toggleFold = function() {\n    return $scope.nodeUnfolded = !$scope.nodeUnfolded;\n  };\n}).controller('JobPlanSubtasksController', function($scope, JobsService) {\n  var getSubtasks;\n  getSubtasks = function() {\n    return JobsService.getSubtasks($scope.nodeid).then(function(data) {\n      return $scope.subtasks = data;\n    });\n  };\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.st)) {\n    getSubtasks();\n  }\n  return $scope.$on('reload', function(event) {\n    if ($scope.nodeid) {\n      return getSubtasks();\n    }\n  });\n}).controller('JobPlanTaskManagersController', function($scope, JobsService) {\n  var getTaskManagers;\n  getTaskManagers = function() {\n    return JobsService.getTaskManagers($scope.nodeid).then(function(data) {\n      return $scope.taskmanagers = data;\n    });\n  };\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.st)) {\n    getTaskManagers();\n  }\n  return $scope.$on('reload', function(event) {\n    if ($scope.nodeid) {\n      return getTaskManagers();\n    }\n  });\n}).controller('JobPlanAccumulatorsController', function($scope, JobsService) {\n  var getAccumulators;\n  getAccumulators = function() {\n    return JobsService.getAccumulators($scope.nodeid).then(function(data) {\n      $scope.accumulators = data.main;\n      return $scope.subtaskAccumulators = data.subtasks;\n    });\n  };\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.accumulators)) {\n    getAccumulators();\n  }\n  return $scope.$on('reload', function(event) {\n    if ($scope.nodeid) {\n      return getAccumulators();\n    }\n  });\n}).controller('JobPlanCheckpointsController', function($scope, JobsService) {\n  var getJobCheckpointStats, getOperatorCheckpointStats;\n  getJobCheckpointStats = function() {\n    return JobsService.getJobCheckpointStats($scope.jobid).then(function(data) {\n      return $scope.jobCheckpointStats = data;\n    });\n  };\n  getOperatorCheckpointStats = function() {\n    return JobsService.getOperatorCheckpointStats($scope.nodeid).then(function(data) {\n      $scope.operatorCheckpointStats = data.operatorStats;\n      return $scope.subtasksCheckpointStats = data.subtasksStats;\n    });\n  };\n  getJobCheckpointStats();\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.operatorCheckpointStats)) {\n    getOperatorCheckpointStats();\n  }\n  return $scope.$on('reload', function(event) {\n    getJobCheckpointStats();\n    if ($scope.nodeid) {\n      return getOperatorCheckpointStats();\n    }\n  });\n}).controller('JobPlanBackPressureController', function($scope, JobsService) {\n  var getOperatorBackPressure;\n  getOperatorBackPressure = function() {\n    $scope.now = Date.now();\n    if ($scope.nodeid) {\n      return JobsService.getOperatorBackPressure($scope.nodeid).then(function(data) {\n        return $scope.backPressureOperatorStats[$scope.nodeid] = data;\n      });\n    }\n  };\n  getOperatorBackPressure();\n  return $scope.$on('reload', function(event) {\n    return getOperatorBackPressure();\n  });\n}).controller('JobTimelineVertexController', function($scope, $state, $stateParams, JobsService) {\n  var getVertex;\n  getVertex = function() {\n    return JobsService.getVertex($stateParams.vertexId).then(function(data) {\n      return $scope.vertex = data;\n    });\n  };\n  getVertex();\n  return $scope.$on('reload', function(event) {\n    return getVertex();\n  });\n}).controller('JobExceptionsController', function($scope, $state, $stateParams, JobsService) {\n  return JobsService.loadExceptions().then(function(data) {\n    return $scope.exceptions = data;\n  });\n}).controller('JobPropertiesController', function($scope, JobsService) {\n  return $scope.changeNode = function(nodeid) {\n    if (nodeid !== $scope.nodeid) {\n      $scope.nodeid = nodeid;\n      return JobsService.getNode(nodeid).then(function(data) {\n        return $scope.node = data;\n      });\n    } else {\n      $scope.nodeid = null;\n      return $scope.node = null;\n    }\n  };\n}).controller('JobPlanMetricsController', function($scope, JobsService, MetricsService) {\n  var loadMetrics;\n  $scope.dragging = false;\n  $scope.window = MetricsService.getWindow();\n  $scope.availableMetrics = null;\n  $scope.$on('$destroy', function() {\n    return MetricsService.unRegisterObserver();\n  });\n  loadMetrics = function() {\n    JobsService.getVertex($scope.nodeid).then(function(data) {\n      return $scope.vertex = data;\n    });\n    return MetricsService.getAvailableMetrics($scope.jobid, $scope.nodeid).then(function(data) {\n      $scope.availableMetrics = data;\n      $scope.metrics = MetricsService.getMetricsSetup($scope.jobid, $scope.nodeid).names;\n      return MetricsService.registerObserver($scope.jobid, $scope.nodeid, function(data) {\n        return $scope.$broadcast(\"metrics:data:update\", data.timestamp, data.values);\n      });\n    });\n  };\n  $scope.dropped = function(event, index, item, external, type) {\n    MetricsService.orderMetrics($scope.jobid, $scope.nodeid, item, index);\n    $scope.$broadcast(\"metrics:refresh\", item);\n    loadMetrics();\n    return false;\n  };\n  $scope.dragStart = function() {\n    return $scope.dragging = true;\n  };\n  $scope.dragEnd = function() {\n    return $scope.dragging = false;\n  };\n  $scope.addMetric = function(metric) {\n    MetricsService.addMetric($scope.jobid, $scope.nodeid, metric.id);\n    return loadMetrics();\n  };\n  $scope.removeMetric = function(metric) {\n    MetricsService.removeMetric($scope.jobid, $scope.nodeid, metric);\n    return loadMetrics();\n  };\n  $scope.setMetricSize = function(metric, size) {\n    MetricsService.setMetricSize($scope.jobid, $scope.nodeid, metric, size);\n    return loadMetrics();\n  };\n  $scope.getValues = function(metric) {\n    return MetricsService.getValues($scope.jobid, $scope.nodeid, metric);\n  };\n  $scope.$on('node:change', function(event, nodeid) {\n    if (!$scope.dragging) {\n      return loadMetrics();\n    }\n  });\n  if ($scope.nodeid) {\n    return loadMetrics();\n  }\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n# ----------------------------------------------\n\n.directive 'vertex', ($state) ->\n  template: \"<svg class='timeline secondary' width='0' height='0'></svg>\"\n\n  scope:\n    data: \"=\"\n\n  link: (scope, elem, attrs) ->\n    svgEl = elem.children()[0]\n\n    containerW = elem.width()\n    angular.element(svgEl).attr('width', containerW)\n\n    analyzeTime = (data) ->\n      d3.select(svgEl).selectAll(\"*\").remove()\n\n      testData = []\n\n      angular.forEach data.subtasks, (subtask, i) ->\n        times = [\n          {\n            label: \"Scheduled\"\n            color: \"#666\"\n            borderColor: \"#555\"\n            starting_time: subtask.timestamps[\"SCHEDULED\"]\n            ending_time: subtask.timestamps[\"DEPLOYING\"]\n            type: 'regular'\n          }\n          {\n            label: \"Deploying\"\n            color: \"#aaa\"\n            borderColor: \"#555\"\n            starting_time: subtask.timestamps[\"DEPLOYING\"]\n            ending_time: subtask.timestamps[\"RUNNING\"]\n            type: 'regular'\n          }\n        ]\n\n        if subtask.timestamps[\"FINISHED\"] > 0\n          times.push {\n            label: \"Running\"\n            color: \"#ddd\"\n            borderColor: \"#555\"\n            starting_time: subtask.timestamps[\"RUNNING\"]\n            ending_time: subtask.timestamps[\"FINISHED\"]\n            type: 'regular'\n          }\n\n        testData.push {\n          label: \"(#{subtask.subtask}) #{subtask.host}\"\n          times: times\n        }\n\n      chart = d3.timeline().stack()\n      .tickFormat({\n        format: d3.time.format(\"%L\")\n        # tickInterval: 1\n        tickSize: 1\n      })\n      .prefix(\"single\")\n      .labelFormat((label) ->\n        label\n      )\n      .margin({ left: 100, right: 0, top: 0, bottom: 0 })\n      .itemHeight(30)\n      .relativeTime()\n\n      svg = d3.select(svgEl)\n      .datum(testData)\n      .call(chart)\n\n    analyzeTime(scope.data)\n\n    return\n\n# ----------------------------------------------\n\n.directive 'timeline', ($state) ->\n  template: \"<svg class='timeline' width='0' height='0'></svg>\"\n\n  scope:\n    vertices: \"=\"\n    jobid: \"=\"\n\n  link: (scope, elem, attrs) ->\n    svgEl = elem.children()[0]\n\n    containerW = elem.width()\n    angular.element(svgEl).attr('width', containerW)\n\n    translateLabel = (label) ->\n      label.replace(\"&gt;\", \">\")\n\n    analyzeTime = (data) ->\n      d3.select(svgEl).selectAll(\"*\").remove()\n\n      testData = []\n\n      angular.forEach data, (vertex) ->\n        if vertex['start-time'] > -1\n          if vertex.type is 'scheduled'\n            testData.push\n              times: [\n                label: translateLabel(vertex.name)\n                color: \"#cccccc\"\n                borderColor: \"#555555\"\n                starting_time: vertex['start-time']\n                ending_time: vertex['end-time']\n                type: vertex.type\n              ]\n          else\n            testData.push\n              times: [\n                label: translateLabel(vertex.name)\n                color: \"#d9f1f7\"\n                borderColor: \"#62cdea\"\n                starting_time: vertex['start-time']\n                ending_time: vertex['end-time']\n                link: vertex.id\n                type: vertex.type\n              ]\n\n      chart = d3.timeline().stack().click((d, i, datum) ->\n        if d.link\n          $state.go \"single-job.timeline.vertex\", { jobid: scope.jobid, vertexId: d.link }\n\n      )\n      .tickFormat({\n        format: d3.time.format(\"%L\")\n        # tickTime: d3.time.second\n        # tickInterval: 0.5\n        tickSize: 1\n      })\n      .prefix(\"main\")\n      .margin({ left: 0, right: 0, top: 0, bottom: 0 })\n      .itemHeight(30)\n      .showBorderLine()\n      .showHourTimeline()\n\n      svg = d3.select(svgEl)\n      .datum(testData)\n      .call(chart)\n\n    scope.$watch attrs.vertices, (data) ->\n      analyzeTime(data) if data\n\n    return\n\n# ----------------------------------------------\n.directive 'split', () -> \n  return compile: (tElem, tAttrs) ->\n      Split(tElem.children(), (\n        sizes: [50, 50]\n        direction: 'vertical'\n      ))\n# ----------------------------------------------\n\n.directive 'jobPlan', ($timeout) ->\n  template: \"\n    <svg class='graph' width='500' height='400'><g /></svg>\n    <svg class='tmp' width='1' height='1'><g /></svg>\n    <div class='btn-group zoom-buttons'>\n      <a class='btn btn-default zoom-in' ng-click='zoomIn()'><i class='fa fa-plus' /></a>\n      <a class='btn btn-default zoom-out' ng-click='zoomOut()'><i class='fa fa-minus' /></a>\n    </div>\"\n\n  scope:\n    plan: '='\n    setNode: '&'\n\n  link: (scope, elem, attrs) ->\n    g = null\n    mainZoom = d3.behavior.zoom()\n    subgraphs = []\n    jobid = attrs.jobid\n\n    mainSvgElement = elem.children()[0]\n    mainG = elem.children().children()[0]\n    mainTmpElement = elem.children()[1]\n\n    d3mainSvg = d3.select(mainSvgElement)\n    d3mainSvgG = d3.select(mainG)\n    d3tmpSvg = d3.select(mainTmpElement)\n\n    # angular.element(mainG).empty()\n    # d3mainSvgG.selectAll(\"*\").remove()\n\n    containerW = elem.width()\n    angular.element(elem.children()[0]).width(containerW)\n\n    scope.zoomIn = ->\n      if mainZoom.scale() < 2.99\n\n        # Calculate and store new values in zoom object\n        translate = mainZoom.translate()\n        v1 = translate[0] * (mainZoom.scale() + 0.1 / (mainZoom.scale()))\n        v2 = translate[1] * (mainZoom.scale() + 0.1 / (mainZoom.scale()))\n        mainZoom.scale mainZoom.scale() + 0.1\n        mainZoom.translate [ v1, v2 ]\n\n        # Transform svg\n        d3mainSvgG.attr \"transform\", \"translate(\" + v1 + \",\" + v2 + \") scale(\" + mainZoom.scale() + \")\"\n\n    scope.zoomOut = ->\n      if mainZoom.scale() > 0.31\n\n        # Calculate and store new values in mainZoom object\n        mainZoom.scale mainZoom.scale() - 0.1\n        translate = mainZoom.translate()\n        v1 = translate[0] * (mainZoom.scale() - 0.1 / (mainZoom.scale()))\n        v2 = translate[1] * (mainZoom.scale() - 0.1 / (mainZoom.scale()))\n        mainZoom.translate [ v1, v2 ]\n\n        # Transform svg\n        d3mainSvgG.attr \"transform\", \"translate(\" + v1 + \",\" + v2 + \") scale(\" + mainZoom.scale() + \")\"\n\n    #create a label of an edge\n    createLabelEdge = (el) ->\n      labelValue = \"\"\n      if el.ship_strategy? or el.local_strategy?\n        labelValue += \"<div class='edge-label'>\"\n        labelValue += el.ship_strategy  if el.ship_strategy?\n        labelValue += \" (\" + el.temp_mode + \")\"  unless el.temp_mode is `undefined`\n        labelValue += \",<br>\" + el.local_strategy  unless el.local_strategy is `undefined`\n        labelValue += \"</div>\"\n      labelValue\n\n\n    # true, if the node is a special node from an iteration\n    isSpecialIterationNode = (info) ->\n      (info is \"partialSolution\" or info is \"nextPartialSolution\" or info is \"workset\" or info is \"nextWorkset\" or info is \"solutionSet\" or info is \"solutionDelta\")\n\n    getNodeType = (el, info) ->\n      if info is \"mirror\"\n        'node-mirror'\n\n      else if isSpecialIterationNode(info)\n        'node-iteration'\n\n      else\n        'node-normal'\n\n    # creates the label of a node, in info is stored, whether it is a special node (like a mirror in an iteration)\n    createLabelNode = (el, info, maxW, maxH) ->\n      # labelValue = \"<a href='#/jobs/\" + jobid + \"/vertex/\" + el.id + \"' class='node-label \" + getNodeType(el, info) + \"'>\"\n      labelValue = \"<div href='#/jobs/\" + jobid + \"/vertex/\" + el.id + \"' class='node-label \" + getNodeType(el, info) + \"'>\"\n\n      # Nodename\n      if info is \"mirror\"\n        labelValue += \"<h3 class='node-name'>Mirror of \" + el.operator + \"</h3>\"\n      else\n        labelValue += \"<h3 class='node-name'>\" + el.operator + \"</h3>\"\n      if el.description is \"\"\n        labelValue += \"\"\n      else\n        stepName = el.description\n\n        # clean stepName\n        stepName = shortenString(stepName)\n        labelValue += \"<h4 class='step-name'>\" + stepName + \"</h4>\"\n\n      # If this node is an \"iteration\" we need a different panel-body\n      if el.step_function?\n        labelValue += extendLabelNodeForIteration(el.id, maxW, maxH)\n      else\n\n        # Otherwise add infos\n        labelValue += \"<h5>\" + info + \" Node</h5>\"  if isSpecialIterationNode(info)\n        labelValue += \"<h5>Parallelism: \" + el.parallelism + \"</h5>\"  unless el.parallelism is \"\"\n        labelValue += \"<h5>Operation: \" + shortenString(el.operator_strategy) + \"</h5>\" unless el.operator is `undefined` or not el.operator_strategy\n      # labelValue += \"</a>\"\n      labelValue += \"</div>\"\n      labelValue\n\n    # Extends the label of a node with an additional svg Element to present the iteration.\n    extendLabelNodeForIteration = (id, maxW, maxH) ->\n      svgID = \"svg-\" + id\n\n      labelValue = \"<svg class='\" + svgID + \"' width=\" + maxW + \" height=\" + maxH + \"><g /></svg>\"\n      labelValue\n\n    # Split a string into multiple lines so that each line has less than 30 letters.\n    shortenString = (s) ->\n      # make sure that name does not contain a < (because of html)\n      if s.charAt(0) is \"<\"\n        s = s.replace(\"<\", \"&lt;\")\n        s = s.replace(\">\", \"&gt;\")\n      sbr = \"\"\n      while s.length > 30\n        sbr = sbr + s.substring(0, 30) + \"<br>\"\n        s = s.substring(30, s.length)\n      sbr = sbr + s\n      sbr\n\n    createNode = (g, data, el, isParent = false, maxW, maxH) ->\n      # create node, send additional informations about the node if it is a special one\n      if el.id is data.partial_solution\n        g.setNode el.id,\n          label: createLabelNode(el, \"partialSolution\", maxW, maxH)\n          labelType: 'html'\n          class: getNodeType(el, \"partialSolution\")\n\n      else if el.id is data.next_partial_solution\n        g.setNode el.id,\n          label: createLabelNode(el, \"nextPartialSolution\", maxW, maxH)\n          labelType: 'html'\n          class: getNodeType(el, \"nextPartialSolution\")\n\n      else if el.id is data.workset\n        g.setNode el.id,\n          label: createLabelNode(el, \"workset\", maxW, maxH)\n          labelType: 'html'\n          class: getNodeType(el, \"workset\")\n\n      else if el.id is data.next_workset\n        g.setNode el.id,\n          label: createLabelNode(el, \"nextWorkset\", maxW, maxH)\n          labelType: 'html'\n          class: getNodeType(el, \"nextWorkset\")\n\n      else if el.id is data.solution_set\n        g.setNode el.id,\n          label: createLabelNode(el, \"solutionSet\", maxW, maxH)\n          labelType: 'html'\n          class: getNodeType(el, \"solutionSet\")\n\n      else if el.id is data.solution_delta\n        g.setNode el.id,\n          label: createLabelNode(el, \"solutionDelta\", maxW, maxH)\n          labelType: 'html'\n          class: getNodeType(el, \"solutionDelta\")\n\n      else\n        g.setNode el.id,\n          label: createLabelNode(el, \"\", maxW, maxH)\n          labelType: 'html'\n          class: getNodeType(el, \"\")\n\n    createEdge = (g, data, el, existingNodes, pred) ->\n      g.setEdge pred.id, el.id,\n        label: createLabelEdge(pred)\n        labelType: 'html'\n        arrowhead: 'normal'\n\n    loadJsonToDagre = (g, data) ->\n      existingNodes = []\n\n      if data.nodes?\n        # This is the normal json data\n        toIterate = data.nodes\n\n      else\n        # This is an iteration, we now store special iteration nodes if possible\n        toIterate = data.step_function\n        isParent = true\n\n      for el in toIterate\n        maxW = 0\n        maxH = 0\n\n        if el.step_function\n          sg = new dagreD3.graphlib.Graph({ multigraph: true, compound: true }).setGraph({\n            nodesep: 20\n            edgesep: 0\n            ranksep: 20\n            rankdir: \"LR\"\n            marginx: 10\n            marginy: 10\n            })\n\n          subgraphs[el.id] = sg\n\n          loadJsonToDagre(sg, el)\n\n          r = new dagreD3.render()\n          d3tmpSvg.select('g').call(r, sg)\n          maxW = sg.graph().width\n          maxH = sg.graph().height\n\n          angular.element(mainTmpElement).empty()\n\n        createNode(g, data, el, isParent, maxW, maxH)\n\n        existingNodes.push el.id\n\n        # create edges from inputs to current node\n        if el.inputs?\n          for pred in el.inputs\n            createEdge(g, data, el, existingNodes, pred)\n\n      g\n\n    # searches in the global JSONData for the node with the given id\n    searchForNode = (data, nodeID) ->\n      for i of data.nodes\n        el = data.nodes[i]\n        return el  if el.id is nodeID\n\n        # look for nodes that are in iterations\n        if el.step_function?\n          for j of el.step_function\n            return el.step_function[j]  if el.step_function[j].id is nodeID\n\n    drawGraph = (data) ->\n      g = new dagreD3.graphlib.Graph({ multigraph: true, compound: true }).setGraph({\n        nodesep: 70\n        edgesep: 0\n        ranksep: 50\n        rankdir: \"LR\"\n        marginx: 40\n        marginy: 40\n        })\n\n      loadJsonToDagre(g, data)\n\n      renderer = new dagreD3.render()\n      d3mainSvgG.call(renderer, g)\n\n      for i, sg of subgraphs\n        d3mainSvg.select('svg.svg-' + i + ' g').call(renderer, sg)\n\n      newScale = 0.5\n\n      xCenterOffset = Math.floor((angular.element(mainSvgElement).width() - g.graph().width * newScale) / 2)\n      yCenterOffset = Math.floor((angular.element(mainSvgElement).height() - g.graph().height * newScale) / 2)\n\n      mainZoom.scale(newScale).translate([xCenterOffset, yCenterOffset])\n\n      d3mainSvgG.attr(\"transform\", \"translate(\" + xCenterOffset + \", \" + yCenterOffset + \") scale(\" + mainZoom.scale() + \")\")\n\n      mainZoom.on(\"zoom\", ->\n        ev = d3.event\n        d3mainSvgG.attr \"transform\", \"translate(\" + ev.translate + \") scale(\" + ev.scale + \")\"\n      )\n      mainZoom(d3mainSvg)\n\n      d3mainSvgG.selectAll('.node').on 'click', (d) ->\n        scope.setNode({ nodeid: d })\n\n    scope.$watch attrs.plan, (newPlan) ->\n      drawGraph(newPlan) if newPlan\n\n    return\n","angular.module('flinkApp').directive('vertex', function($state) {\n  return {\n    template: \"<svg class='timeline secondary' width='0' height='0'></svg>\",\n    scope: {\n      data: \"=\"\n    },\n    link: function(scope, elem, attrs) {\n      var analyzeTime, containerW, svgEl;\n      svgEl = elem.children()[0];\n      containerW = elem.width();\n      angular.element(svgEl).attr('width', containerW);\n      analyzeTime = function(data) {\n        var chart, svg, testData;\n        d3.select(svgEl).selectAll(\"*\").remove();\n        testData = [];\n        angular.forEach(data.subtasks, function(subtask, i) {\n          var times;\n          times = [\n            {\n              label: \"Scheduled\",\n              color: \"#666\",\n              borderColor: \"#555\",\n              starting_time: subtask.timestamps[\"SCHEDULED\"],\n              ending_time: subtask.timestamps[\"DEPLOYING\"],\n              type: 'regular'\n            }, {\n              label: \"Deploying\",\n              color: \"#aaa\",\n              borderColor: \"#555\",\n              starting_time: subtask.timestamps[\"DEPLOYING\"],\n              ending_time: subtask.timestamps[\"RUNNING\"],\n              type: 'regular'\n            }\n          ];\n          if (subtask.timestamps[\"FINISHED\"] > 0) {\n            times.push({\n              label: \"Running\",\n              color: \"#ddd\",\n              borderColor: \"#555\",\n              starting_time: subtask.timestamps[\"RUNNING\"],\n              ending_time: subtask.timestamps[\"FINISHED\"],\n              type: 'regular'\n            });\n          }\n          return testData.push({\n            label: \"(\" + subtask.subtask + \") \" + subtask.host,\n            times: times\n          });\n        });\n        chart = d3.timeline().stack().tickFormat({\n          format: d3.time.format(\"%L\"),\n          tickSize: 1\n        }).prefix(\"single\").labelFormat(function(label) {\n          return label;\n        }).margin({\n          left: 100,\n          right: 0,\n          top: 0,\n          bottom: 0\n        }).itemHeight(30).relativeTime();\n        return svg = d3.select(svgEl).datum(testData).call(chart);\n      };\n      analyzeTime(scope.data);\n    }\n  };\n}).directive('timeline', function($state) {\n  return {\n    template: \"<svg class='timeline' width='0' height='0'></svg>\",\n    scope: {\n      vertices: \"=\",\n      jobid: \"=\"\n    },\n    link: function(scope, elem, attrs) {\n      var analyzeTime, containerW, svgEl, translateLabel;\n      svgEl = elem.children()[0];\n      containerW = elem.width();\n      angular.element(svgEl).attr('width', containerW);\n      translateLabel = function(label) {\n        return label.replace(\"&gt;\", \">\");\n      };\n      analyzeTime = function(data) {\n        var chart, svg, testData;\n        d3.select(svgEl).selectAll(\"*\").remove();\n        testData = [];\n        angular.forEach(data, function(vertex) {\n          if (vertex['start-time'] > -1) {\n            if (vertex.type === 'scheduled') {\n              return testData.push({\n                times: [\n                  {\n                    label: translateLabel(vertex.name),\n                    color: \"#cccccc\",\n                    borderColor: \"#555555\",\n                    starting_time: vertex['start-time'],\n                    ending_time: vertex['end-time'],\n                    type: vertex.type\n                  }\n                ]\n              });\n            } else {\n              return testData.push({\n                times: [\n                  {\n                    label: translateLabel(vertex.name),\n                    color: \"#d9f1f7\",\n                    borderColor: \"#62cdea\",\n                    starting_time: vertex['start-time'],\n                    ending_time: vertex['end-time'],\n                    link: vertex.id,\n                    type: vertex.type\n                  }\n                ]\n              });\n            }\n          }\n        });\n        chart = d3.timeline().stack().click(function(d, i, datum) {\n          if (d.link) {\n            return $state.go(\"single-job.timeline.vertex\", {\n              jobid: scope.jobid,\n              vertexId: d.link\n            });\n          }\n        }).tickFormat({\n          format: d3.time.format(\"%L\"),\n          tickSize: 1\n        }).prefix(\"main\").margin({\n          left: 0,\n          right: 0,\n          top: 0,\n          bottom: 0\n        }).itemHeight(30).showBorderLine().showHourTimeline();\n        return svg = d3.select(svgEl).datum(testData).call(chart);\n      };\n      scope.$watch(attrs.vertices, function(data) {\n        if (data) {\n          return analyzeTime(data);\n        }\n      });\n    }\n  };\n}).directive('split', function() {\n  return {\n    compile: function(tElem, tAttrs) {\n      return Split(tElem.children(), {\n        sizes: [50, 50],\n        direction: 'vertical'\n      });\n    }\n  };\n}).directive('jobPlan', function($timeout) {\n  return {\n    template: \"<svg class='graph' width='500' height='400'><g /></svg> <svg class='tmp' width='1' height='1'><g /></svg> <div class='btn-group zoom-buttons'> <a class='btn btn-default zoom-in' ng-click='zoomIn()'><i class='fa fa-plus' /></a> <a class='btn btn-default zoom-out' ng-click='zoomOut()'><i class='fa fa-minus' /></a> </div>\",\n    scope: {\n      plan: '=',\n      setNode: '&'\n    },\n    link: function(scope, elem, attrs) {\n      var containerW, createEdge, createLabelEdge, createLabelNode, createNode, d3mainSvg, d3mainSvgG, d3tmpSvg, drawGraph, extendLabelNodeForIteration, g, getNodeType, isSpecialIterationNode, jobid, loadJsonToDagre, mainG, mainSvgElement, mainTmpElement, mainZoom, searchForNode, shortenString, subgraphs;\n      g = null;\n      mainZoom = d3.behavior.zoom();\n      subgraphs = [];\n      jobid = attrs.jobid;\n      mainSvgElement = elem.children()[0];\n      mainG = elem.children().children()[0];\n      mainTmpElement = elem.children()[1];\n      d3mainSvg = d3.select(mainSvgElement);\n      d3mainSvgG = d3.select(mainG);\n      d3tmpSvg = d3.select(mainTmpElement);\n      containerW = elem.width();\n      angular.element(elem.children()[0]).width(containerW);\n      scope.zoomIn = function() {\n        var translate, v1, v2;\n        if (mainZoom.scale() < 2.99) {\n          translate = mainZoom.translate();\n          v1 = translate[0] * (mainZoom.scale() + 0.1 / (mainZoom.scale()));\n          v2 = translate[1] * (mainZoom.scale() + 0.1 / (mainZoom.scale()));\n          mainZoom.scale(mainZoom.scale() + 0.1);\n          mainZoom.translate([v1, v2]);\n          return d3mainSvgG.attr(\"transform\", \"translate(\" + v1 + \",\" + v2 + \") scale(\" + mainZoom.scale() + \")\");\n        }\n      };\n      scope.zoomOut = function() {\n        var translate, v1, v2;\n        if (mainZoom.scale() > 0.31) {\n          mainZoom.scale(mainZoom.scale() - 0.1);\n          translate = mainZoom.translate();\n          v1 = translate[0] * (mainZoom.scale() - 0.1 / (mainZoom.scale()));\n          v2 = translate[1] * (mainZoom.scale() - 0.1 / (mainZoom.scale()));\n          mainZoom.translate([v1, v2]);\n          return d3mainSvgG.attr(\"transform\", \"translate(\" + v1 + \",\" + v2 + \") scale(\" + mainZoom.scale() + \")\");\n        }\n      };\n      createLabelEdge = function(el) {\n        var labelValue;\n        labelValue = \"\";\n        if ((el.ship_strategy != null) || (el.local_strategy != null)) {\n          labelValue += \"<div class='edge-label'>\";\n          if (el.ship_strategy != null) {\n            labelValue += el.ship_strategy;\n          }\n          if (el.temp_mode !== undefined) {\n            labelValue += \" (\" + el.temp_mode + \")\";\n          }\n          if (el.local_strategy !== undefined) {\n            labelValue += \",<br>\" + el.local_strategy;\n          }\n          labelValue += \"</div>\";\n        }\n        return labelValue;\n      };\n      isSpecialIterationNode = function(info) {\n        return info === \"partialSolution\" || info === \"nextPartialSolution\" || info === \"workset\" || info === \"nextWorkset\" || info === \"solutionSet\" || info === \"solutionDelta\";\n      };\n      getNodeType = function(el, info) {\n        if (info === \"mirror\") {\n          return 'node-mirror';\n        } else if (isSpecialIterationNode(info)) {\n          return 'node-iteration';\n        } else {\n          return 'node-normal';\n        }\n      };\n      createLabelNode = function(el, info, maxW, maxH) {\n        var labelValue, stepName;\n        labelValue = \"<div href='#/jobs/\" + jobid + \"/vertex/\" + el.id + \"' class='node-label \" + getNodeType(el, info) + \"'>\";\n        if (info === \"mirror\") {\n          labelValue += \"<h3 class='node-name'>Mirror of \" + el.operator + \"</h3>\";\n        } else {\n          labelValue += \"<h3 class='node-name'>\" + el.operator + \"</h3>\";\n        }\n        if (el.description === \"\") {\n          labelValue += \"\";\n        } else {\n          stepName = el.description;\n          stepName = shortenString(stepName);\n          labelValue += \"<h4 class='step-name'>\" + stepName + \"</h4>\";\n        }\n        if (el.step_function != null) {\n          labelValue += extendLabelNodeForIteration(el.id, maxW, maxH);\n        } else {\n          if (isSpecialIterationNode(info)) {\n            labelValue += \"<h5>\" + info + \" Node</h5>\";\n          }\n          if (el.parallelism !== \"\") {\n            labelValue += \"<h5>Parallelism: \" + el.parallelism + \"</h5>\";\n          }\n          if (!(el.operator === undefined || !el.operator_strategy)) {\n            labelValue += \"<h5>Operation: \" + shortenString(el.operator_strategy) + \"</h5>\";\n          }\n        }\n        labelValue += \"</div>\";\n        return labelValue;\n      };\n      extendLabelNodeForIteration = function(id, maxW, maxH) {\n        var labelValue, svgID;\n        svgID = \"svg-\" + id;\n        labelValue = \"<svg class='\" + svgID + \"' width=\" + maxW + \" height=\" + maxH + \"><g /></svg>\";\n        return labelValue;\n      };\n      shortenString = function(s) {\n        var sbr;\n        if (s.charAt(0) === \"<\") {\n          s = s.replace(\"<\", \"&lt;\");\n          s = s.replace(\">\", \"&gt;\");\n        }\n        sbr = \"\";\n        while (s.length > 30) {\n          sbr = sbr + s.substring(0, 30) + \"<br>\";\n          s = s.substring(30, s.length);\n        }\n        sbr = sbr + s;\n        return sbr;\n      };\n      createNode = function(g, data, el, isParent, maxW, maxH) {\n        if (isParent == null) {\n          isParent = false;\n        }\n        if (el.id === data.partial_solution) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"partialSolution\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"partialSolution\")\n          });\n        } else if (el.id === data.next_partial_solution) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"nextPartialSolution\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"nextPartialSolution\")\n          });\n        } else if (el.id === data.workset) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"workset\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"workset\")\n          });\n        } else if (el.id === data.next_workset) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"nextWorkset\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"nextWorkset\")\n          });\n        } else if (el.id === data.solution_set) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"solutionSet\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"solutionSet\")\n          });\n        } else if (el.id === data.solution_delta) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"solutionDelta\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"solutionDelta\")\n          });\n        } else {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"\")\n          });\n        }\n      };\n      createEdge = function(g, data, el, existingNodes, pred) {\n        return g.setEdge(pred.id, el.id, {\n          label: createLabelEdge(pred),\n          labelType: 'html',\n          arrowhead: 'normal'\n        });\n      };\n      loadJsonToDagre = function(g, data) {\n        var el, existingNodes, isParent, k, l, len, len1, maxH, maxW, pred, r, ref, sg, toIterate;\n        existingNodes = [];\n        if (data.nodes != null) {\n          toIterate = data.nodes;\n        } else {\n          toIterate = data.step_function;\n          isParent = true;\n        }\n        for (k = 0, len = toIterate.length; k < len; k++) {\n          el = toIterate[k];\n          maxW = 0;\n          maxH = 0;\n          if (el.step_function) {\n            sg = new dagreD3.graphlib.Graph({\n              multigraph: true,\n              compound: true\n            }).setGraph({\n              nodesep: 20,\n              edgesep: 0,\n              ranksep: 20,\n              rankdir: \"LR\",\n              marginx: 10,\n              marginy: 10\n            });\n            subgraphs[el.id] = sg;\n            loadJsonToDagre(sg, el);\n            r = new dagreD3.render();\n            d3tmpSvg.select('g').call(r, sg);\n            maxW = sg.graph().width;\n            maxH = sg.graph().height;\n            angular.element(mainTmpElement).empty();\n          }\n          createNode(g, data, el, isParent, maxW, maxH);\n          existingNodes.push(el.id);\n          if (el.inputs != null) {\n            ref = el.inputs;\n            for (l = 0, len1 = ref.length; l < len1; l++) {\n              pred = ref[l];\n              createEdge(g, data, el, existingNodes, pred);\n            }\n          }\n        }\n        return g;\n      };\n      searchForNode = function(data, nodeID) {\n        var el, i, j;\n        for (i in data.nodes) {\n          el = data.nodes[i];\n          if (el.id === nodeID) {\n            return el;\n          }\n          if (el.step_function != null) {\n            for (j in el.step_function) {\n              if (el.step_function[j].id === nodeID) {\n                return el.step_function[j];\n              }\n            }\n          }\n        }\n      };\n      drawGraph = function(data) {\n        var i, newScale, renderer, sg, xCenterOffset, yCenterOffset;\n        g = new dagreD3.graphlib.Graph({\n          multigraph: true,\n          compound: true\n        }).setGraph({\n          nodesep: 70,\n          edgesep: 0,\n          ranksep: 50,\n          rankdir: \"LR\",\n          marginx: 40,\n          marginy: 40\n        });\n        loadJsonToDagre(g, data);\n        renderer = new dagreD3.render();\n        d3mainSvgG.call(renderer, g);\n        for (i in subgraphs) {\n          sg = subgraphs[i];\n          d3mainSvg.select('svg.svg-' + i + ' g').call(renderer, sg);\n        }\n        newScale = 0.5;\n        xCenterOffset = Math.floor((angular.element(mainSvgElement).width() - g.graph().width * newScale) / 2);\n        yCenterOffset = Math.floor((angular.element(mainSvgElement).height() - g.graph().height * newScale) / 2);\n        mainZoom.scale(newScale).translate([xCenterOffset, yCenterOffset]);\n        d3mainSvgG.attr(\"transform\", \"translate(\" + xCenterOffset + \", \" + yCenterOffset + \") scale(\" + mainZoom.scale() + \")\");\n        mainZoom.on(\"zoom\", function() {\n          var ev;\n          ev = d3.event;\n          return d3mainSvgG.attr(\"transform\", \"translate(\" + ev.translate + \") scale(\" + ev.scale + \")\");\n        });\n        mainZoom(d3mainSvg);\n        return d3mainSvgG.selectAll('.node').on('click', function(d) {\n          return scope.setNode({\n            nodeid: d\n          });\n        });\n      };\n      scope.$watch(attrs.plan, function(newPlan) {\n        if (newPlan) {\n          return drawGraph(newPlan);\n        }\n      });\n    }\n  };\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.service 'JobsService', ($http, flinkConfig, $log, amMoment, $q, $timeout) ->\n  currentJob = null\n  currentPlan = null\n\n  deferreds = {}\n  jobs = {\n    running: []\n    finished: []\n    cancelled: []\n    failed: []\n  }\n\n  jobObservers = []\n\n  notifyObservers = ->\n    angular.forEach jobObservers, (callback) ->\n      callback()\n\n  @registerObserver = (callback) ->\n    jobObservers.push(callback)\n\n  @unRegisterObserver = (callback) ->\n    index = jobObservers.indexOf(callback)\n    jobObservers.splice(index, 1)\n\n  @stateList = ->\n    [ \n      # 'CREATED'\n      'SCHEDULED'\n      'DEPLOYING'\n      'RUNNING'\n      'FINISHED'\n      'FAILED'\n      'CANCELING'\n      'CANCELED'\n    ]\n\n  @translateLabelState = (state) ->\n    switch state.toLowerCase()\n      when 'finished' then 'success'\n      when 'failed' then 'danger'\n      when 'scheduled' then 'default'\n      when 'deploying' then 'info'\n      when 'running' then 'primary'\n      when 'canceling' then 'warning'\n      when 'pending' then 'info'\n      when 'total' then 'black'\n      else 'default'\n\n  @setEndTimes = (list) ->\n    angular.forEach list, (item, jobKey) ->\n      unless item['end-time'] > -1\n        item['end-time'] = item['start-time'] + item['duration']\n\n  @processVertices = (data) ->\n    angular.forEach data.vertices, (vertex, i) ->\n      vertex.type = 'regular'\n\n    data.vertices.unshift({\n      name: 'Scheduled'\n      'start-time': data.timestamps['CREATED']\n      'end-time': data.timestamps['CREATED'] + 1\n      type: 'scheduled'\n    })\n\n  @listJobs = ->\n    deferred = $q.defer()\n\n    $http.get flinkConfig.jobServer + \"joboverview\"\n    .success (data, status, headers, config) =>\n      angular.forEach data, (list, listKey) =>\n        switch listKey\n          when 'running' then jobs.running = @setEndTimes(list)\n          when 'finished' then jobs.finished = @setEndTimes(list)\n          when 'cancelled' then jobs.cancelled = @setEndTimes(list)\n          when 'failed' then jobs.failed = @setEndTimes(list)\n\n      deferred.resolve(jobs)\n      notifyObservers()\n\n    deferred.promise\n\n  @getJobs = (type) ->\n    jobs[type]\n\n  @getAllJobs = ->\n    jobs\n\n  @loadJob = (jobid) ->\n    currentJob = null\n    deferreds.job = $q.defer()\n\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid\n    .success (data, status, headers, config) =>\n      @setEndTimes(data.vertices)\n      @processVertices(data)\n\n      $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/config\"\n      .success (jobConfig) ->\n        data = angular.extend(data, jobConfig)\n\n        currentJob = data\n\n        deferreds.job.resolve(currentJob)\n\n    deferreds.job.promise\n\n  @getNode = (nodeid) ->\n    seekNode = (nodeid, data) ->\n      for node in data\n        return node if node.id is nodeid\n        sub = seekNode(nodeid, node.step_function) if node.step_function\n        return sub if sub\n\n      null\n\n    deferred = $q.defer()\n\n    deferreds.job.promise.then (data) =>\n      foundNode = seekNode(nodeid, currentJob.plan.nodes)\n\n      foundNode.vertex = @seekVertex(nodeid)\n\n      deferred.resolve(foundNode)\n\n    deferred.promise\n\n  @seekVertex = (nodeid) ->\n    for vertex in currentJob.vertices\n      return vertex if vertex.id is nodeid\n\n    return null\n\n  @getVertex = (vertexid) ->\n    deferred = $q.defer()\n\n    deferreds.job.promise.then (data) =>\n      vertex = @seekVertex(vertexid)\n\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/subtasktimes\"\n      .success (data) =>\n        # TODO: change to subtasktimes\n        vertex.subtasks = data.subtasks\n\n        deferred.resolve(vertex)\n\n    deferred.promise\n\n  @getSubtasks = (vertexid) ->\n    deferred = $q.defer()\n\n    deferreds.job.promise.then (data) =>\n      # vertex = @seekVertex(vertexid)\n\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid\n      .success (data) ->\n        subtasks = data.subtasks\n\n        deferred.resolve(subtasks)\n\n    deferred.promise\n\n  @getTaskManagers = (vertexid) ->\n    deferred = $q.defer()\n\n    deferreds.job.promise.then (data) =>\n      # vertex = @seekVertex(vertexid)\n\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/taskmanagers\"\n      .success (data) ->\n        taskmanagers = data.taskmanagers\n\n        deferred.resolve(taskmanagers)\n\n    deferred.promise\n\n  @getAccumulators = (vertexid) ->\n    deferred = $q.defer()\n\n    deferreds.job.promise.then (data) =>\n      # vertex = @seekVertex(vertexid)\n\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/accumulators\"\n      .success (data) ->\n        accumulators = data['user-accumulators']\n\n        $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/subtasks/accumulators\"\n        .success (data) ->\n          subtaskAccumulators = data.subtasks\n\n          deferred.resolve({ main: accumulators, subtasks: subtaskAccumulators })\n\n    deferred.promise\n\n  # Job-level checkpoint stats\n  @getJobCheckpointStats = (jobid) ->\n    deferred = $q.defer()\n\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/checkpoints\"\n    .success (data, status, headers, config) =>\n      if (angular.equals({}, data))\n        deferred.resolve(deferred.resolve(null))\n      else\n        deferred.resolve(data)\n\n    deferred.promise\n\n  # Operator-level checkpoint stats\n  @getOperatorCheckpointStats = (vertexid) ->\n    deferred = $q.defer()\n\n    deferreds.job.promise.then (data) =>\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/checkpoints\"\n      .success (data) ->\n        # If no data available, we are done.\n        if (angular.equals({}, data))\n          deferred.resolve({ operatorStats: null, subtasksStats: null })\n        else\n          operatorStats = { id: data['id'], timestamp: data['timestamp'], duration: data['duration'], size: data['size'] }\n\n          if (angular.equals({}, data['subtasks']))\n            deferred.resolve({ operatorStats: operatorStats, subtasksStats: null })\n          else\n            subtaskStats = data['subtasks']\n            deferred.resolve({ operatorStats: operatorStats, subtasksStats: subtaskStats })\n\n    deferred.promise\n\n  # Operator-level back pressure stats\n  @getOperatorBackPressure = (vertexid) ->\n    deferred = $q.defer()\n\n    $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/backpressure\"\n    .success (data) =>\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @translateBackPressureLabelState = (state) ->\n    switch state.toLowerCase()\n      when 'in-progress' then 'danger'\n      when 'ok' then 'success'\n      when 'low' then 'warning'\n      when 'high' then 'danger'\n      else 'default'\n\n  @loadExceptions = ->\n    deferred = $q.defer()\n\n    deferreds.job.promise.then (data) =>\n\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/exceptions\"\n      .success (exceptions) ->\n        currentJob.exceptions = exceptions\n\n        deferred.resolve(exceptions)\n\n    deferred.promise\n\n  @cancelJob = (jobid) ->\n    # uses the non REST-compliant GET yarn-cancel handler which is available in addition to the\n    # proper \"DELETE jobs/<jobid>/\"\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/yarn-cancel\"\n\n  @stopJob = (jobid) ->\n    # uses the non REST-compliant GET yarn-cancel handler which is available in addition to the\n    # proper \"DELETE jobs/<jobid>/\"\n    $http.get \"jobs/\" + jobid + \"/yarn-stop\"\n\n  @\n","angular.module('flinkApp').service('JobsService', function($http, flinkConfig, $log, amMoment, $q, $timeout) {\n  var currentJob, currentPlan, deferreds, jobObservers, jobs, notifyObservers;\n  currentJob = null;\n  currentPlan = null;\n  deferreds = {};\n  jobs = {\n    running: [],\n    finished: [],\n    cancelled: [],\n    failed: []\n  };\n  jobObservers = [];\n  notifyObservers = function() {\n    return angular.forEach(jobObservers, function(callback) {\n      return callback();\n    });\n  };\n  this.registerObserver = function(callback) {\n    return jobObservers.push(callback);\n  };\n  this.unRegisterObserver = function(callback) {\n    var index;\n    index = jobObservers.indexOf(callback);\n    return jobObservers.splice(index, 1);\n  };\n  this.stateList = function() {\n    return ['SCHEDULED', 'DEPLOYING', 'RUNNING', 'FINISHED', 'FAILED', 'CANCELING', 'CANCELED'];\n  };\n  this.translateLabelState = function(state) {\n    switch (state.toLowerCase()) {\n      case 'finished':\n        return 'success';\n      case 'failed':\n        return 'danger';\n      case 'scheduled':\n        return 'default';\n      case 'deploying':\n        return 'info';\n      case 'running':\n        return 'primary';\n      case 'canceling':\n        return 'warning';\n      case 'pending':\n        return 'info';\n      case 'total':\n        return 'black';\n      default:\n        return 'default';\n    }\n  };\n  this.setEndTimes = function(list) {\n    return angular.forEach(list, function(item, jobKey) {\n      if (!(item['end-time'] > -1)) {\n        return item['end-time'] = item['start-time'] + item['duration'];\n      }\n    });\n  };\n  this.processVertices = function(data) {\n    angular.forEach(data.vertices, function(vertex, i) {\n      return vertex.type = 'regular';\n    });\n    return data.vertices.unshift({\n      name: 'Scheduled',\n      'start-time': data.timestamps['CREATED'],\n      'end-time': data.timestamps['CREATED'] + 1,\n      type: 'scheduled'\n    });\n  };\n  this.listJobs = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"joboverview\").success((function(_this) {\n      return function(data, status, headers, config) {\n        angular.forEach(data, function(list, listKey) {\n          switch (listKey) {\n            case 'running':\n              return jobs.running = _this.setEndTimes(list);\n            case 'finished':\n              return jobs.finished = _this.setEndTimes(list);\n            case 'cancelled':\n              return jobs.cancelled = _this.setEndTimes(list);\n            case 'failed':\n              return jobs.failed = _this.setEndTimes(list);\n          }\n        });\n        deferred.resolve(jobs);\n        return notifyObservers();\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getJobs = function(type) {\n    return jobs[type];\n  };\n  this.getAllJobs = function() {\n    return jobs;\n  };\n  this.loadJob = function(jobid) {\n    currentJob = null;\n    deferreds.job = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobs/\" + jobid).success((function(_this) {\n      return function(data, status, headers, config) {\n        _this.setEndTimes(data.vertices);\n        _this.processVertices(data);\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/config\").success(function(jobConfig) {\n          data = angular.extend(data, jobConfig);\n          currentJob = data;\n          return deferreds.job.resolve(currentJob);\n        });\n      };\n    })(this));\n    return deferreds.job.promise;\n  };\n  this.getNode = function(nodeid) {\n    var deferred, seekNode;\n    seekNode = function(nodeid, data) {\n      var j, len, node, sub;\n      for (j = 0, len = data.length; j < len; j++) {\n        node = data[j];\n        if (node.id === nodeid) {\n          return node;\n        }\n        if (node.step_function) {\n          sub = seekNode(nodeid, node.step_function);\n        }\n        if (sub) {\n          return sub;\n        }\n      }\n      return null;\n    };\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        var foundNode;\n        foundNode = seekNode(nodeid, currentJob.plan.nodes);\n        foundNode.vertex = _this.seekVertex(nodeid);\n        return deferred.resolve(foundNode);\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.seekVertex = function(nodeid) {\n    var j, len, ref, vertex;\n    ref = currentJob.vertices;\n    for (j = 0, len = ref.length; j < len; j++) {\n      vertex = ref[j];\n      if (vertex.id === nodeid) {\n        return vertex;\n      }\n    }\n    return null;\n  };\n  this.getVertex = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        var vertex;\n        vertex = _this.seekVertex(vertexid);\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/subtasktimes\").success(function(data) {\n          vertex.subtasks = data.subtasks;\n          return deferred.resolve(vertex);\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getSubtasks = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid).success(function(data) {\n          var subtasks;\n          subtasks = data.subtasks;\n          return deferred.resolve(subtasks);\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getTaskManagers = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/taskmanagers\").success(function(data) {\n          var taskmanagers;\n          taskmanagers = data.taskmanagers;\n          return deferred.resolve(taskmanagers);\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getAccumulators = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/accumulators\").success(function(data) {\n          var accumulators;\n          accumulators = data['user-accumulators'];\n          return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/subtasks/accumulators\").success(function(data) {\n            var subtaskAccumulators;\n            subtaskAccumulators = data.subtasks;\n            return deferred.resolve({\n              main: accumulators,\n              subtasks: subtaskAccumulators\n            });\n          });\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getJobCheckpointStats = function(jobid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/checkpoints\").success((function(_this) {\n      return function(data, status, headers, config) {\n        if (angular.equals({}, data)) {\n          return deferred.resolve(deferred.resolve(null));\n        } else {\n          return deferred.resolve(data);\n        }\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getOperatorCheckpointStats = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/checkpoints\").success(function(data) {\n          var operatorStats, subtaskStats;\n          if (angular.equals({}, data)) {\n            return deferred.resolve({\n              operatorStats: null,\n              subtasksStats: null\n            });\n          } else {\n            operatorStats = {\n              id: data['id'],\n              timestamp: data['timestamp'],\n              duration: data['duration'],\n              size: data['size']\n            };\n            if (angular.equals({}, data['subtasks'])) {\n              return deferred.resolve({\n                operatorStats: operatorStats,\n                subtasksStats: null\n              });\n            } else {\n              subtaskStats = data['subtasks'];\n              return deferred.resolve({\n                operatorStats: operatorStats,\n                subtasksStats: subtaskStats\n              });\n            }\n          }\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getOperatorBackPressure = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/backpressure\").success((function(_this) {\n      return function(data) {\n        return deferred.resolve(data);\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.translateBackPressureLabelState = function(state) {\n    switch (state.toLowerCase()) {\n      case 'in-progress':\n        return 'danger';\n      case 'ok':\n        return 'success';\n      case 'low':\n        return 'warning';\n      case 'high':\n        return 'danger';\n      default:\n        return 'default';\n    }\n  };\n  this.loadExceptions = function() {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/exceptions\").success(function(exceptions) {\n          currentJob.exceptions = exceptions;\n          return deferred.resolve(exceptions);\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.cancelJob = function(jobid) {\n    return $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/yarn-cancel\");\n  };\n  this.stopJob = function(jobid) {\n    return $http.get(\"jobs/\" + jobid + \"/yarn-stop\");\n  };\n  return this;\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n# ----------------------------------------------\n\n.directive 'metricsGraph', ->\n  template: '<div class=\"panel panel-default panel-metric\">\n               <div class=\"panel-heading\">\n                 <span class=\"metric-title\">{{metric.id}}</span>\n                 <div class=\"buttons\">\n                   <div class=\"btn-group\">\n                     <button type=\"button\" ng-class=\"[btnClasses, {active: metric.size != \\'big\\'}]\" ng-click=\"setSize(\\'small\\')\">Small</button>\n                     <button type=\"button\" ng-class=\"[btnClasses, {active: metric.size == \\'big\\'}]\" ng-click=\"setSize(\\'big\\')\">Big</button>\n                   </div>\n                   <a title=\"Remove\" class=\"btn btn-default btn-xs remove\" ng-click=\"removeMetric()\"><i class=\"fa fa-close\" /></a>\n                 </div>\n               </div>\n               <div class=\"panel-body\">\n                  <svg />\n               </div>\n             </div>'\n  replace: true\n  scope:\n    metric: \"=\"\n    window: \"=\"\n    removeMetric: \"&\"\n    setMetricSize: \"=\"\n    getValues: \"&\"\n\n  link: (scope, element, attrs) ->\n    scope.btnClasses = ['btn', 'btn-default', 'btn-xs']\n\n    scope.value = null\n    scope.data = [{\n      values: scope.getValues()\n    }]\n\n    scope.options = {\n      x: (d, i) ->\n        d.x\n      y: (d, i) ->\n        d.y\n\n      xTickFormat: (d) ->\n        d3.time.format('%H:%M:%S')(new Date(d))\n\n      yTickFormat: (d) ->\n        found = false\n        pow = 0\n        step = 1\n        absD = Math.abs(d)\n\n        while !found && pow < 50\n          if Math.pow(10, pow) <= absD && absD < Math.pow(10, pow + step)\n            found = true\n          else\n            pow += step\n\n        if found && pow > 6\n          \"#{d / Math.pow(10, pow)}E#{pow}\"\n        else\n          \"#{d}\"\n    }\n\n    scope.showChart = ->\n      d3.select(element.find(\"svg\")[0])\n      .datum(scope.data)\n      .transition().duration(250)\n      .call(scope.chart)\n\n    scope.chart = nv.models.lineChart()\n      .options(scope.options)\n      .showLegend(false)\n      .margin({\n        top: 15\n        left: 60\n        bottom: 30\n        right: 30\n      })\n\n    scope.chart.yAxis.showMaxMin(false)\n    scope.chart.tooltip.hideDelay(0)\n    scope.chart.tooltip.contentGenerator((obj) ->\n      \"<p>#{d3.time.format('%H:%M:%S')(new Date(obj.point.x))} | #{obj.point.y}</p>\"\n    )\n\n    nv.utils.windowResize(scope.chart.update);\n\n    scope.setSize = (size) ->\n      scope.setMetricSize(scope.metric, size)\n\n    scope.showChart()\n\n    scope.$on 'metrics:data:update', (event, timestamp, data) ->\n#      scope.value = parseInt(data[scope.metric.id])\n      scope.value = parseFloat(data[scope.metric.id])\n\n      scope.data[0].values.push {\n        x: timestamp\n        y: scope.value\n      }\n\n      if scope.data[0].values.length > scope.window\n        scope.data[0].values.shift()\n\n      scope.showChart()\n      scope.chart.clearHighlights()\n      scope.chart.tooltip.hidden(true)\n\n    element.find(\".metric-title\").qtip({\n      content: {\n        text: scope.metric.id\n      },\n      position: {\n        my: 'bottom left',\n        at: 'top left'\n      },\n      style: {\n        classes: 'qtip-light qtip-timeline-bar'\n      }\n    });\n","angular.module('flinkApp').directive('metricsGraph', function() {\n  return {\n    template: '<div class=\"panel panel-default panel-metric\"> <div class=\"panel-heading\"> <span class=\"metric-title\">{{metric.id}}</span> <div class=\"buttons\"> <div class=\"btn-group\"> <button type=\"button\" ng-class=\"[btnClasses, {active: metric.size != \\'big\\'}]\" ng-click=\"setSize(\\'small\\')\">Small</button> <button type=\"button\" ng-class=\"[btnClasses, {active: metric.size == \\'big\\'}]\" ng-click=\"setSize(\\'big\\')\">Big</button> </div> <a title=\"Remove\" class=\"btn btn-default btn-xs remove\" ng-click=\"removeMetric()\"><i class=\"fa fa-close\" /></a> </div> </div> <div class=\"panel-body\"> <svg /> </div> </div>',\n    replace: true,\n    scope: {\n      metric: \"=\",\n      window: \"=\",\n      removeMetric: \"&\",\n      setMetricSize: \"=\",\n      getValues: \"&\"\n    },\n    link: function(scope, element, attrs) {\n      scope.btnClasses = ['btn', 'btn-default', 'btn-xs'];\n      scope.value = null;\n      scope.data = [\n        {\n          values: scope.getValues()\n        }\n      ];\n      scope.options = {\n        x: function(d, i) {\n          return d.x;\n        },\n        y: function(d, i) {\n          return d.y;\n        },\n        xTickFormat: function(d) {\n          return d3.time.format('%H:%M:%S')(new Date(d));\n        },\n        yTickFormat: function(d) {\n          var absD, found, pow, step;\n          found = false;\n          pow = 0;\n          step = 1;\n          absD = Math.abs(d);\n          while (!found && pow < 50) {\n            if (Math.pow(10, pow) <= absD && absD < Math.pow(10, pow + step)) {\n              found = true;\n            } else {\n              pow += step;\n            }\n          }\n          if (found && pow > 6) {\n            return (d / Math.pow(10, pow)) + \"E\" + pow;\n          } else {\n            return \"\" + d;\n          }\n        }\n      };\n      scope.showChart = function() {\n        return d3.select(element.find(\"svg\")[0]).datum(scope.data).transition().duration(250).call(scope.chart);\n      };\n      scope.chart = nv.models.lineChart().options(scope.options).showLegend(false).margin({\n        top: 15,\n        left: 60,\n        bottom: 30,\n        right: 30\n      });\n      scope.chart.yAxis.showMaxMin(false);\n      scope.chart.tooltip.hideDelay(0);\n      scope.chart.tooltip.contentGenerator(function(obj) {\n        return \"<p>\" + (d3.time.format('%H:%M:%S')(new Date(obj.point.x))) + \" | \" + obj.point.y + \"</p>\";\n      });\n      nv.utils.windowResize(scope.chart.update);\n      scope.setSize = function(size) {\n        return scope.setMetricSize(scope.metric, size);\n      };\n      scope.showChart();\n      scope.$on('metrics:data:update', function(event, timestamp, data) {\n        scope.value = parseFloat(data[scope.metric.id]);\n        scope.data[0].values.push({\n          x: timestamp,\n          y: scope.value\n        });\n        if (scope.data[0].values.length > scope.window) {\n          scope.data[0].values.shift();\n        }\n        scope.showChart();\n        scope.chart.clearHighlights();\n        return scope.chart.tooltip.hidden(true);\n      });\n      return element.find(\".metric-title\").qtip({\n        content: {\n          text: scope.metric.id\n        },\n        position: {\n          my: 'bottom left',\n          at: 'top left'\n        },\n        style: {\n          classes: 'qtip-light qtip-timeline-bar'\n        }\n      });\n    }\n  };\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.service 'MetricsService', ($http, $q, flinkConfig, $interval) ->\n  @metrics = {}\n  @values = {}\n  @watched = {}\n  @observer = {\n    jobid: null\n    nodeid: null\n    callback: null\n  }\n\n  @refresh = $interval =>\n    angular.forEach @watched, (v, jobid) =>\n      angular.forEach v, (nodeid, nk) =>\n        @getAllAvailableMetrics(jobid, nodeid).then (data) =>\n          names = []\n          angular.forEach data, (metric, mk) =>\n            names.push metric.id\n\n          if names.length > 0\n            @getMetrics(jobid, nodeid, names).then (values) =>\n              if jobid == @observer.jobid && nodeid == @observer.nodeid\n                @observer.callback(values) if @observer.callback\n\n\n  , flinkConfig[\"refresh-interval\"]\n\n  @registerObserver = (jobid, nodeid, callback) ->\n    @observer.jobid = jobid\n    @observer.nodeid = nodeid\n    @observer.callback = callback\n\n  @unRegisterObserver = ->\n    @observer = {\n      jobid: null\n      nodeid: null\n      callback: null\n    }\n\n  @setupMetrics = (jobid, vertices) ->\n    @setupLS()\n\n    @watched[jobid] = []\n    angular.forEach vertices, (v, k) =>\n      @watched[jobid].push(v.id) if v.id\n\n  @getWindow = ->\n    100\n\n  @setupLS = ->\n    if !localStorage.flinkMetrics?\n      @saveSetup()\n\n    @metrics = JSON.parse(localStorage.flinkMetrics)\n\n  @saveSetup = ->\n    localStorage.flinkMetrics = JSON.stringify(@metrics)\n\n  @saveValue = (jobid, nodeid, value) ->\n    unless @values[jobid]?\n      @values[jobid] = {}\n\n    unless @values[jobid][nodeid]?\n      @values[jobid][nodeid] = []\n\n    @values[jobid][nodeid].push(value)\n\n    if @values[jobid][nodeid].length > @getWindow()\n      @values[jobid][nodeid].shift()\n\n  @getValues = (jobid, nodeid, metricid) ->\n    return [] unless @values[jobid]?\n    return [] unless @values[jobid][nodeid]?\n\n    results = []\n    angular.forEach @values[jobid][nodeid], (v, k) =>\n      if v.values[metricid]?\n        results.push {\n          x: v.timestamp\n          y: v.values[metricid]\n        }\n\n    results\n\n  @setupLSFor = (jobid, nodeid) ->\n    if !@metrics[jobid]?\n      @metrics[jobid] = {}\n\n    if !@metrics[jobid][nodeid]?\n      @metrics[jobid][nodeid] = []\n\n  @addMetric = (jobid, nodeid, metricid) ->\n    @setupLSFor(jobid, nodeid)\n\n    @metrics[jobid][nodeid].push({id: metricid, size: 'small'})\n\n    @saveSetup()\n\n  @removeMetric = (jobid, nodeid, metric) =>\n    if @metrics[jobid][nodeid]?\n      i = @metrics[jobid][nodeid].indexOf(metric)\n      i = _.findIndex(@metrics[jobid][nodeid], { id: metric }) if i == -1\n\n      @metrics[jobid][nodeid].splice(i, 1) if i != -1\n\n      @saveSetup()\n\n  @setMetricSize = (jobid, nodeid, metric, size) =>\n    if @metrics[jobid][nodeid]?\n      i = @metrics[jobid][nodeid].indexOf(metric.id)\n      i = _.findIndex(@metrics[jobid][nodeid], { id: metric.id }) if i == -1\n\n      @metrics[jobid][nodeid][i] = { id: metric.id, size: size } if i != -1\n\n      @saveSetup()\n\n  @orderMetrics = (jobid, nodeid, item, index) ->\n    @setupLSFor(jobid, nodeid)\n\n    angular.forEach @metrics[jobid][nodeid], (v, k) =>\n      if v.id == item.id\n        @metrics[jobid][nodeid].splice(k, 1)\n        if k < index\n          index = index - 1\n\n    @metrics[jobid][nodeid].splice(index, 0, item)\n\n    @saveSetup()\n\n  @getMetricsSetup = (jobid, nodeid) =>\n    {\n      names: _.map(@metrics[jobid][nodeid], (value) =>\n        if _.isString(value) then { id: value, size: \"small\" } else value\n      )\n    }\n\n  @getAvailableMetrics = (jobid, nodeid) =>\n    @setupLSFor(jobid, nodeid)\n\n    deferred = $q.defer()\n\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics\"\n    .success (data) =>\n      results = []\n      angular.forEach data, (v, k) =>\n        i = @metrics[jobid][nodeid].indexOf(v.id)\n        i = _.findIndex(@metrics[jobid][nodeid], { id: v.id }) if i == -1\n\n        if i == -1\n          results.push(v)\n\n      deferred.resolve(results)\n\n    deferred.promise\n\n  @getAllAvailableMetrics = (jobid, nodeid) =>\n    deferred = $q.defer()\n\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics\"\n    .success (data) =>\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @getMetrics = (jobid, nodeid, metricIds) ->\n    deferred = $q.defer()\n\n    ids = metricIds.join(\",\")\n\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics?get=\" + ids\n    .success (data) =>\n      result = {}\n      angular.forEach data, (v, k) ->\n        result[v.id] = parseInt(v.value)\n\n      newValue = {\n        timestamp: Date.now()\n        values: result\n      }\n      @saveValue(jobid, nodeid, newValue)\n      deferred.resolve(newValue)\n\n    deferred.promise\n\n  @setupLS()\n\n  @\n","angular.module('flinkApp').service('MetricsService', function($http, $q, flinkConfig, $interval) {\n  this.metrics = {};\n  this.values = {};\n  this.watched = {};\n  this.observer = {\n    jobid: null,\n    nodeid: null,\n    callback: null\n  };\n  this.refresh = $interval((function(_this) {\n    return function() {\n      return angular.forEach(_this.watched, function(v, jobid) {\n        return angular.forEach(v, function(nodeid, nk) {\n          return _this.getAllAvailableMetrics(jobid, nodeid).then(function(data) {\n            var names;\n            names = [];\n            angular.forEach(data, function(metric, mk) {\n              return names.push(metric.id);\n            });\n            if (names.length > 0) {\n              return _this.getMetrics(jobid, nodeid, names).then(function(values) {\n                if (jobid === _this.observer.jobid && nodeid === _this.observer.nodeid) {\n                  if (_this.observer.callback) {\n                    return _this.observer.callback(values);\n                  }\n                }\n              });\n            }\n          });\n        });\n      });\n    };\n  })(this), flinkConfig[\"refresh-interval\"]);\n  this.registerObserver = function(jobid, nodeid, callback) {\n    this.observer.jobid = jobid;\n    this.observer.nodeid = nodeid;\n    return this.observer.callback = callback;\n  };\n  this.unRegisterObserver = function() {\n    return this.observer = {\n      jobid: null,\n      nodeid: null,\n      callback: null\n    };\n  };\n  this.setupMetrics = function(jobid, vertices) {\n    this.setupLS();\n    this.watched[jobid] = [];\n    return angular.forEach(vertices, (function(_this) {\n      return function(v, k) {\n        if (v.id) {\n          return _this.watched[jobid].push(v.id);\n        }\n      };\n    })(this));\n  };\n  this.getWindow = function() {\n    return 100;\n  };\n  this.setupLS = function() {\n    if (localStorage.flinkMetrics == null) {\n      this.saveSetup();\n    }\n    return this.metrics = JSON.parse(localStorage.flinkMetrics);\n  };\n  this.saveSetup = function() {\n    return localStorage.flinkMetrics = JSON.stringify(this.metrics);\n  };\n  this.saveValue = function(jobid, nodeid, value) {\n    if (this.values[jobid] == null) {\n      this.values[jobid] = {};\n    }\n    if (this.values[jobid][nodeid] == null) {\n      this.values[jobid][nodeid] = [];\n    }\n    this.values[jobid][nodeid].push(value);\n    if (this.values[jobid][nodeid].length > this.getWindow()) {\n      return this.values[jobid][nodeid].shift();\n    }\n  };\n  this.getValues = function(jobid, nodeid, metricid) {\n    var results;\n    if (this.values[jobid] == null) {\n      return [];\n    }\n    if (this.values[jobid][nodeid] == null) {\n      return [];\n    }\n    results = [];\n    angular.forEach(this.values[jobid][nodeid], (function(_this) {\n      return function(v, k) {\n        if (v.values[metricid] != null) {\n          return results.push({\n            x: v.timestamp,\n            y: v.values[metricid]\n          });\n        }\n      };\n    })(this));\n    return results;\n  };\n  this.setupLSFor = function(jobid, nodeid) {\n    if (this.metrics[jobid] == null) {\n      this.metrics[jobid] = {};\n    }\n    if (this.metrics[jobid][nodeid] == null) {\n      return this.metrics[jobid][nodeid] = [];\n    }\n  };\n  this.addMetric = function(jobid, nodeid, metricid) {\n    this.setupLSFor(jobid, nodeid);\n    this.metrics[jobid][nodeid].push({\n      id: metricid,\n      size: 'small'\n    });\n    return this.saveSetup();\n  };\n  this.removeMetric = (function(_this) {\n    return function(jobid, nodeid, metric) {\n      var i;\n      if (_this.metrics[jobid][nodeid] != null) {\n        i = _this.metrics[jobid][nodeid].indexOf(metric);\n        if (i === -1) {\n          i = _.findIndex(_this.metrics[jobid][nodeid], {\n            id: metric\n          });\n        }\n        if (i !== -1) {\n          _this.metrics[jobid][nodeid].splice(i, 1);\n        }\n        return _this.saveSetup();\n      }\n    };\n  })(this);\n  this.setMetricSize = (function(_this) {\n    return function(jobid, nodeid, metric, size) {\n      var i;\n      if (_this.metrics[jobid][nodeid] != null) {\n        i = _this.metrics[jobid][nodeid].indexOf(metric.id);\n        if (i === -1) {\n          i = _.findIndex(_this.metrics[jobid][nodeid], {\n            id: metric.id\n          });\n        }\n        if (i !== -1) {\n          _this.metrics[jobid][nodeid][i] = {\n            id: metric.id,\n            size: size\n          };\n        }\n        return _this.saveSetup();\n      }\n    };\n  })(this);\n  this.orderMetrics = function(jobid, nodeid, item, index) {\n    this.setupLSFor(jobid, nodeid);\n    angular.forEach(this.metrics[jobid][nodeid], (function(_this) {\n      return function(v, k) {\n        if (v.id === item.id) {\n          _this.metrics[jobid][nodeid].splice(k, 1);\n          if (k < index) {\n            return index = index - 1;\n          }\n        }\n      };\n    })(this));\n    this.metrics[jobid][nodeid].splice(index, 0, item);\n    return this.saveSetup();\n  };\n  this.getMetricsSetup = (function(_this) {\n    return function(jobid, nodeid) {\n      return {\n        names: _.map(_this.metrics[jobid][nodeid], function(value) {\n          if (_.isString(value)) {\n            return {\n              id: value,\n              size: \"small\"\n            };\n          } else {\n            return value;\n          }\n        })\n      };\n    };\n  })(this);\n  this.getAvailableMetrics = (function(_this) {\n    return function(jobid, nodeid) {\n      var deferred;\n      _this.setupLSFor(jobid, nodeid);\n      deferred = $q.defer();\n      $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics\").success(function(data) {\n        var results;\n        results = [];\n        angular.forEach(data, function(v, k) {\n          var i;\n          i = _this.metrics[jobid][nodeid].indexOf(v.id);\n          if (i === -1) {\n            i = _.findIndex(_this.metrics[jobid][nodeid], {\n              id: v.id\n            });\n          }\n          if (i === -1) {\n            return results.push(v);\n          }\n        });\n        return deferred.resolve(results);\n      });\n      return deferred.promise;\n    };\n  })(this);\n  this.getAllAvailableMetrics = (function(_this) {\n    return function(jobid, nodeid) {\n      var deferred;\n      deferred = $q.defer();\n      $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics\").success(function(data) {\n        return deferred.resolve(data);\n      });\n      return deferred.promise;\n    };\n  })(this);\n  this.getMetrics = function(jobid, nodeid, metricIds) {\n    var deferred, ids;\n    deferred = $q.defer();\n    ids = metricIds.join(\",\");\n    $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics?get=\" + ids).success((function(_this) {\n      return function(data) {\n        var newValue, result;\n        result = {};\n        angular.forEach(data, function(v, k) {\n          return result[v.id] = parseInt(v.value);\n        });\n        newValue = {\n          timestamp: Date.now(),\n          values: result\n        };\n        _this.saveValue(jobid, nodeid, newValue);\n        return deferred.resolve(newValue);\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.setupLS();\n  return this;\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.controller 'OverviewController', ($scope, OverviewService, JobsService, $interval, flinkConfig) ->\n  $scope.jobObserver = ->\n    $scope.runningJobs = JobsService.getJobs('running')\n    $scope.finishedJobs = JobsService.getJobs('finished')\n\n  JobsService.registerObserver($scope.jobObserver)\n  $scope.$on '$destroy', ->\n    JobsService.unRegisterObserver($scope.jobObserver)\n\n  $scope.jobObserver()\n\n  OverviewService.loadOverview().then (data) ->\n    $scope.overview = data\n\n  refresh = $interval ->\n    OverviewService.loadOverview().then (data) ->\n      $scope.overview = data\n  , flinkConfig[\"refresh-interval\"]\n\n  $scope.$on '$destroy', ->\n    $interval.cancel(refresh)\n","angular.module('flinkApp').controller('OverviewController', function($scope, OverviewService, JobsService, $interval, flinkConfig) {\n  var refresh;\n  $scope.jobObserver = function() {\n    $scope.runningJobs = JobsService.getJobs('running');\n    return $scope.finishedJobs = JobsService.getJobs('finished');\n  };\n  JobsService.registerObserver($scope.jobObserver);\n  $scope.$on('$destroy', function() {\n    return JobsService.unRegisterObserver($scope.jobObserver);\n  });\n  $scope.jobObserver();\n  OverviewService.loadOverview().then(function(data) {\n    return $scope.overview = data;\n  });\n  refresh = $interval(function() {\n    return OverviewService.loadOverview().then(function(data) {\n      return $scope.overview = data;\n    });\n  }, flinkConfig[\"refresh-interval\"]);\n  return $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.service 'OverviewService', ($http, flinkConfig, $q) ->\n  overview = {}\n\n  @loadOverview = ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"overview\")\n    .success (data, status, headers, config) ->\n      overview = data\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @\n","angular.module('flinkApp').service('OverviewService', function($http, flinkConfig, $q) {\n  var overview;\n  overview = {};\n  this.loadOverview = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"overview\").success(function(data, status, headers, config) {\n      overview = data;\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.controller 'JobSubmitController', ($scope, JobSubmitService, $interval, flinkConfig, $state, $location) ->\n  $scope.yarn = $location.absUrl().indexOf(\"/proxy/application_\") != -1\n  $scope.loadList = () ->\n    JobSubmitService.loadJarList().then (data) ->\n      $scope.address = data.address\n      $scope.noaccess = data.error\n      $scope.jars = data.files\n\n  $scope.defaultState = () ->\n    $scope.plan = null\n    $scope.error = null\n    $scope.state = {\n      selected: null,\n      parallelism: \"\",\n      savepointPath: \"\",\n      allowNonRestoredState: false\n      'entry-class': \"\",\n      'program-args': \"\",\n      'plan-button': \"Show Plan\",\n      'submit-button': \"Submit\",\n      'action-time': 0\n    }\n\n  $scope.defaultState()\n  $scope.uploader = {}\n  $scope.loadList()\n\n  refresh = $interval ->\n    $scope.loadList()\n  , flinkConfig[\"refresh-interval\"]\n\n  $scope.$on '$destroy', ->\n    $interval.cancel(refresh)\n\n  $scope.selectJar = (id) ->\n    if $scope.state.selected == id\n      $scope.defaultState()\n    else\n      $scope.defaultState()\n      $scope.state.selected = id\n\n  $scope.deleteJar = (event, id) ->\n    if $scope.state.selected == id\n      $scope.defaultState()\n    angular.element(event.currentTarget).removeClass(\"fa-remove\").addClass(\"fa-spin fa-spinner\")\n    JobSubmitService.deleteJar(id).then (data) ->\n      angular.element(event.currentTarget).removeClass(\"fa-spin fa-spinner\").addClass(\"fa-remove\")\n      if data.error?\n        alert(data.error)\n\n  $scope.loadEntryClass = (name) ->\n    $scope.state['entry-class'] = name\n\n  $scope.getPlan = () ->\n    if $scope.state['plan-button'] == \"Show Plan\"\n      action = new Date().getTime()\n      $scope.state['action-time'] = action\n      $scope.state['submit-button'] = \"Submit\"\n      $scope.state['plan-button'] = \"Getting Plan\"\n      $scope.error = null\n      $scope.plan = null\n      JobSubmitService.getPlan(\n        $scope.state.selected, {\n          'entry-class': $scope.state['entry-class'],\n          parallelism: $scope.state.parallelism,\n          'program-args': $scope.state['program-args']\n        }\n      ).then (data) ->\n        if action == $scope.state['action-time']\n          $scope.state['plan-button'] = \"Show Plan\"\n          $scope.error = data.error\n          $scope.plan = data.plan\n\n  $scope.runJob = () ->\n    if $scope.state['submit-button'] == \"Submit\"\n      action = new Date().getTime()\n      $scope.state['action-time'] = action\n      $scope.state['submit-button'] = \"Submitting\"\n      $scope.state['plan-button'] = \"Show Plan\"\n      $scope.error = null\n      JobSubmitService.runJob(\n        $scope.state.selected, {\n          'entry-class': $scope.state['entry-class'],\n          parallelism: $scope.state.parallelism,\n          'program-args': $scope.state['program-args'],\n          savepointPath: $scope.state['savepointPath'],\n          allowNonRestoredState: $scope.state['allowNonRestoredState']\n        }\n      ).then (data) ->\n        if action == $scope.state['action-time']\n          $scope.state['submit-button'] = \"Submit\"\n          $scope.error = data.error\n          if data.jobid?\n            $state.go(\"single-job.plan.subtasks\", {jobid: data.jobid})\n\n  # job plan display related stuff\n  $scope.nodeid = null\n  $scope.changeNode = (nodeid) ->\n    if nodeid != $scope.nodeid\n      $scope.nodeid = nodeid\n      $scope.vertex = null\n      $scope.subtasks = null\n      $scope.accumulators = null\n\n      $scope.$broadcast 'reload'\n\n    else\n      $scope.nodeid = null\n      $scope.nodeUnfolded = false\n      $scope.vertex = null\n      $scope.subtasks = null\n      $scope.accumulators = null\n\n  $scope.clearFiles = () ->\n    $scope.uploader = {}\n\n  $scope.uploadFiles = (files) ->\n    # make sure everything is clear again.\n    $scope.uploader = {}\n    if files.length == 1\n      $scope.uploader['file'] = files[0]\n      $scope.uploader['upload'] = true\n    else\n      $scope.uploader['error'] = \"Did ya forget to select a file?\"\n\n  $scope.startUpload = () ->\n    if $scope.uploader['file']?\n      formdata = new FormData()\n      formdata.append(\"jarfile\", $scope.uploader['file'])\n      $scope.uploader['upload'] = false\n      $scope.uploader['success'] = \"Initializing upload...\"\n      xhr = new XMLHttpRequest()\n      xhr.upload.onprogress = (event) ->\n        $scope.uploader['success'] = null\n        $scope.uploader['progress'] = parseInt(100 * event.loaded / event.total)\n      xhr.upload.onerror = (event) ->\n        $scope.uploader['progress'] = null\n        $scope.uploader['error'] = \"An error occurred while uploading your file\"\n      xhr.upload.onload = (event) ->\n        $scope.uploader['progress'] = null\n        $scope.uploader['success'] = \"Saving...\"\n      xhr.onreadystatechange = () ->\n        if xhr.readyState == 4\n          response = JSON.parse(xhr.responseText)\n          if response.error?\n            $scope.uploader['error'] = response.error\n            $scope.uploader['success'] = null\n          else\n            $scope.uploader['success'] = \"Uploaded!\"\n      xhr.open(\"POST\", \"/jars/upload\")\n      xhr.send(formdata)\n    else\n      console.log(\"Unexpected Error. This should not happen\")\n\n.filter 'getJarSelectClass', ->\n  (selected, actual) ->\n    if selected == actual\n      \"fa-check-square\"\n    else\n      \"fa-square-o\"\n","angular.module('flinkApp').controller('JobSubmitController', function($scope, JobSubmitService, $interval, flinkConfig, $state, $location) {\n  var refresh;\n  $scope.yarn = $location.absUrl().indexOf(\"/proxy/application_\") !== -1;\n  $scope.loadList = function() {\n    return JobSubmitService.loadJarList().then(function(data) {\n      $scope.address = data.address;\n      $scope.noaccess = data.error;\n      return $scope.jars = data.files;\n    });\n  };\n  $scope.defaultState = function() {\n    $scope.plan = null;\n    $scope.error = null;\n    return $scope.state = {\n      selected: null,\n      parallelism: \"\",\n      savepointPath: \"\",\n      allowNonRestoredState: false,\n      'entry-class': \"\",\n      'program-args': \"\",\n      'plan-button': \"Show Plan\",\n      'submit-button': \"Submit\",\n      'action-time': 0\n    };\n  };\n  $scope.defaultState();\n  $scope.uploader = {};\n  $scope.loadList();\n  refresh = $interval(function() {\n    return $scope.loadList();\n  }, flinkConfig[\"refresh-interval\"]);\n  $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n  $scope.selectJar = function(id) {\n    if ($scope.state.selected === id) {\n      return $scope.defaultState();\n    } else {\n      $scope.defaultState();\n      return $scope.state.selected = id;\n    }\n  };\n  $scope.deleteJar = function(event, id) {\n    if ($scope.state.selected === id) {\n      $scope.defaultState();\n    }\n    angular.element(event.currentTarget).removeClass(\"fa-remove\").addClass(\"fa-spin fa-spinner\");\n    return JobSubmitService.deleteJar(id).then(function(data) {\n      angular.element(event.currentTarget).removeClass(\"fa-spin fa-spinner\").addClass(\"fa-remove\");\n      if (data.error != null) {\n        return alert(data.error);\n      }\n    });\n  };\n  $scope.loadEntryClass = function(name) {\n    return $scope.state['entry-class'] = name;\n  };\n  $scope.getPlan = function() {\n    var action;\n    if ($scope.state['plan-button'] === \"Show Plan\") {\n      action = new Date().getTime();\n      $scope.state['action-time'] = action;\n      $scope.state['submit-button'] = \"Submit\";\n      $scope.state['plan-button'] = \"Getting Plan\";\n      $scope.error = null;\n      $scope.plan = null;\n      return JobSubmitService.getPlan($scope.state.selected, {\n        'entry-class': $scope.state['entry-class'],\n        parallelism: $scope.state.parallelism,\n        'program-args': $scope.state['program-args']\n      }).then(function(data) {\n        if (action === $scope.state['action-time']) {\n          $scope.state['plan-button'] = \"Show Plan\";\n          $scope.error = data.error;\n          return $scope.plan = data.plan;\n        }\n      });\n    }\n  };\n  $scope.runJob = function() {\n    var action;\n    if ($scope.state['submit-button'] === \"Submit\") {\n      action = new Date().getTime();\n      $scope.state['action-time'] = action;\n      $scope.state['submit-button'] = \"Submitting\";\n      $scope.state['plan-button'] = \"Show Plan\";\n      $scope.error = null;\n      return JobSubmitService.runJob($scope.state.selected, {\n        'entry-class': $scope.state['entry-class'],\n        parallelism: $scope.state.parallelism,\n        'program-args': $scope.state['program-args'],\n        savepointPath: $scope.state['savepointPath'],\n        allowNonRestoredState: $scope.state['allowNonRestoredState']\n      }).then(function(data) {\n        if (action === $scope.state['action-time']) {\n          $scope.state['submit-button'] = \"Submit\";\n          $scope.error = data.error;\n          if (data.jobid != null) {\n            return $state.go(\"single-job.plan.subtasks\", {\n              jobid: data.jobid\n            });\n          }\n        }\n      });\n    }\n  };\n  $scope.nodeid = null;\n  $scope.changeNode = function(nodeid) {\n    if (nodeid !== $scope.nodeid) {\n      $scope.nodeid = nodeid;\n      $scope.vertex = null;\n      $scope.subtasks = null;\n      $scope.accumulators = null;\n      return $scope.$broadcast('reload');\n    } else {\n      $scope.nodeid = null;\n      $scope.nodeUnfolded = false;\n      $scope.vertex = null;\n      $scope.subtasks = null;\n      return $scope.accumulators = null;\n    }\n  };\n  $scope.clearFiles = function() {\n    return $scope.uploader = {};\n  };\n  $scope.uploadFiles = function(files) {\n    $scope.uploader = {};\n    if (files.length === 1) {\n      $scope.uploader['file'] = files[0];\n      return $scope.uploader['upload'] = true;\n    } else {\n      return $scope.uploader['error'] = \"Did ya forget to select a file?\";\n    }\n  };\n  return $scope.startUpload = function() {\n    var formdata, xhr;\n    if ($scope.uploader['file'] != null) {\n      formdata = new FormData();\n      formdata.append(\"jarfile\", $scope.uploader['file']);\n      $scope.uploader['upload'] = false;\n      $scope.uploader['success'] = \"Initializing upload...\";\n      xhr = new XMLHttpRequest();\n      xhr.upload.onprogress = function(event) {\n        $scope.uploader['success'] = null;\n        return $scope.uploader['progress'] = parseInt(100 * event.loaded / event.total);\n      };\n      xhr.upload.onerror = function(event) {\n        $scope.uploader['progress'] = null;\n        return $scope.uploader['error'] = \"An error occurred while uploading your file\";\n      };\n      xhr.upload.onload = function(event) {\n        $scope.uploader['progress'] = null;\n        return $scope.uploader['success'] = \"Saving...\";\n      };\n      xhr.onreadystatechange = function() {\n        var response;\n        if (xhr.readyState === 4) {\n          response = JSON.parse(xhr.responseText);\n          if (response.error != null) {\n            $scope.uploader['error'] = response.error;\n            return $scope.uploader['success'] = null;\n          } else {\n            return $scope.uploader['success'] = \"Uploaded!\";\n          }\n        }\n      };\n      xhr.open(\"POST\", \"/jars/upload\");\n      return xhr.send(formdata);\n    } else {\n      return console.log(\"Unexpected Error. This should not happen\");\n    }\n  };\n}).filter('getJarSelectClass', function() {\n  return function(selected, actual) {\n    if (selected === actual) {\n      return \"fa-check-square\";\n    } else {\n      return \"fa-square-o\";\n    }\n  };\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.service 'JobSubmitService', ($http, flinkConfig, $q) ->\n\n  @loadJarList = () ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"jars/\")\n    .success (data, status, headers, config) ->\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @deleteJar = (id) ->\n    deferred = $q.defer()\n\n    $http.delete(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id))\n    .success (data, status, headers, config) ->\n       deferred.resolve(data)\n\n    deferred.promise\n\n  @getPlan = (id, args) ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id) + \"/plan\", {params: args})\n    .success (data, status, headers, config) ->\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @runJob = (id, args) ->\n    deferred = $q.defer()\n\n    $http.post(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id) + \"/run\", {}, {params: args})\n    .success (data, status, headers, config) ->\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @\n","angular.module('flinkApp').service('JobSubmitService', function($http, flinkConfig, $q) {\n  this.loadJarList = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jars/\").success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  this.deleteJar = function(id) {\n    var deferred;\n    deferred = $q.defer();\n    $http[\"delete\"](flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id)).success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  this.getPlan = function(id, args) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id) + \"/plan\", {\n      params: args\n    }).success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  this.runJob = function(id, args) {\n    var deferred;\n    deferred = $q.defer();\n    $http.post(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id) + \"/run\", {}, {\n      params: args\n    }).success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.controller 'AllTaskManagersController', ($scope, TaskManagersService, $interval, flinkConfig) ->\n  TaskManagersService.loadManagers().then (data) ->\n    $scope.managers = data\n\n  refresh = $interval ->\n    TaskManagersService.loadManagers().then (data) ->\n      $scope.managers = data\n  , flinkConfig[\"refresh-interval\"]\n\n  $scope.$on '$destroy', ->\n    $interval.cancel(refresh)\n\n.controller 'SingleTaskManagerController', ($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) ->\n  $scope.metrics = {}\n  SingleTaskManagerService.loadMetrics($stateParams.taskmanagerid).then (data) ->\n      $scope.metrics = data[0]\n\n    refresh = $interval ->\n      SingleTaskManagerService.loadMetrics($stateParams.taskmanagerid).then (data) ->\n        $scope.metrics = data[0]\n    , flinkConfig[\"refresh-interval\"]\n\n    $scope.$on '$destroy', ->\n      $interval.cancel(refresh)\n\n.controller 'SingleTaskManagerLogsController', ($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) ->\n  $scope.log = {}\n  SingleTaskManagerService.loadLogs($stateParams.taskmanagerid).then (data) ->\n    $scope.log = data\n\n  $scope.reloadData = () ->\n    SingleTaskManagerService.loadLogs($stateParams.taskmanagerid).then (data) ->\n      $scope.log = data\n\n  $scope.downloadData = () ->\n    window.location.href = \"/taskmanagers/\" + ($stateParams.taskmanagerid) + \"/log\"\n\n.controller 'SingleTaskManagerStdoutController', ($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) ->\n  $scope.stdout = {}\n  SingleTaskManagerService.loadStdout($stateParams.taskmanagerid).then (data) ->\n    $scope.stdout = data\n\n  $scope.reloadData = () ->\n    SingleTaskManagerService.loadStdout($stateParams.taskmanagerid).then (data) ->\n      $scope.stdout = data\n\n  $scope.downloadData = () ->\n    window.location.href = \"/taskmanagers/\" + ($stateParams.taskmanagerid) + \"/stdout\"\n","angular.module('flinkApp').controller('AllTaskManagersController', function($scope, TaskManagersService, $interval, flinkConfig) {\n  var refresh;\n  TaskManagersService.loadManagers().then(function(data) {\n    return $scope.managers = data;\n  });\n  refresh = $interval(function() {\n    return TaskManagersService.loadManagers().then(function(data) {\n      return $scope.managers = data;\n    });\n  }, flinkConfig[\"refresh-interval\"]);\n  return $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n}).controller('SingleTaskManagerController', function($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) {\n  var refresh;\n  $scope.metrics = {};\n  SingleTaskManagerService.loadMetrics($stateParams.taskmanagerid).then(function(data) {\n    return $scope.metrics = data[0];\n  });\n  refresh = $interval(function() {\n    return SingleTaskManagerService.loadMetrics($stateParams.taskmanagerid).then(function(data) {\n      return $scope.metrics = data[0];\n    });\n  }, flinkConfig[\"refresh-interval\"]);\n  return $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n}).controller('SingleTaskManagerLogsController', function($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) {\n  $scope.log = {};\n  SingleTaskManagerService.loadLogs($stateParams.taskmanagerid).then(function(data) {\n    return $scope.log = data;\n  });\n  $scope.reloadData = function() {\n    return SingleTaskManagerService.loadLogs($stateParams.taskmanagerid).then(function(data) {\n      return $scope.log = data;\n    });\n  };\n  return $scope.downloadData = function() {\n    return window.location.href = \"/taskmanagers/\" + $stateParams.taskmanagerid + \"/log\";\n  };\n}).controller('SingleTaskManagerStdoutController', function($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) {\n  $scope.stdout = {};\n  SingleTaskManagerService.loadStdout($stateParams.taskmanagerid).then(function(data) {\n    return $scope.stdout = data;\n  });\n  $scope.reloadData = function() {\n    return SingleTaskManagerService.loadStdout($stateParams.taskmanagerid).then(function(data) {\n      return $scope.stdout = data;\n    });\n  };\n  return $scope.downloadData = function() {\n    return window.location.href = \"/taskmanagers/\" + $stateParams.taskmanagerid + \"/stdout\";\n  };\n});\n","#\n# Licensed to the Apache Software Foundation (ASF) under one\n# or more contributor license agreements.  See the NOTICE file\n# distributed with this work for additional information\n# regarding copyright ownership.  The ASF licenses this file\n# to you under the Apache License, Version 2.0 (the\n# \"License\"); you may not use this file except in compliance\n# with the License.  You may obtain a copy of the License at\n#\n#     http://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\nangular.module('flinkApp')\n\n.service 'TaskManagersService', ($http, flinkConfig, $q) ->\n  @loadManagers = () ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"taskmanagers\")\n    .success (data, status, headers, config) ->\n      deferred.resolve(data['taskmanagers'])\n\n    deferred.promise\n\n  @\n\n.service 'SingleTaskManagerService', ($http, flinkConfig, $q) ->\n  @loadMetrics = (taskmanagerid) ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/metrics\")\n    .success (data, status, headers, config) ->\n      deferred.resolve(data['taskmanagers'])\n\n    deferred.promise\n\n  @loadLogs = (taskmanagerid) ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/log\")\n    .success (data, status, headers, config) ->\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @loadStdout = (taskmanagerid) ->\n    deferred = $q.defer()\n\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/stdout\")\n    .success (data, status, headers, config) ->\n      deferred.resolve(data)\n\n    deferred.promise\n\n  @\n\n","angular.module('flinkApp').service('TaskManagersService', function($http, flinkConfig, $q) {\n  this.loadManagers = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"taskmanagers\").success(function(data, status, headers, config) {\n      return deferred.resolve(data['taskmanagers']);\n    });\n    return deferred.promise;\n  };\n  return this;\n}).service('SingleTaskManagerService', function($http, flinkConfig, $q) {\n  this.loadMetrics = function(taskmanagerid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/metrics\").success(function(data, status, headers, config) {\n      return deferred.resolve(data['taskmanagers']);\n    });\n    return deferred.promise;\n  };\n  this.loadLogs = function(taskmanagerid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/log\").success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  this.loadStdout = function(taskmanagerid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/stdout\").success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n"]} +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["index.coffee","index.js","common/directives.coffee","common/directives.js","common/filters.coffee","common/filters.js","common/services.coffee","common/services.js","modules/jobmanager/jobmanager.ctrl.coffee","modules/jobmanager/jobmanager.ctrl.js","modules/jobmanager/jobmanager.svc.coffee","modules/jobmanager/jobmanager.svc.js","modules/jobs/jobs.ctrl.coffee","modules/jobs/jobs.ctrl.js","modules/jobs/jobs.dir.coffee","modules/jobs/jobs.dir.js","modules/jobs/jobs.svc.coffee","modules/jobs/jobs.svc.js","modules/jobs/metrics.dir.coffee","modules/jobs/metrics.dir.js","modules/jobs/metrics.svc.coffee","modules/jobs/metrics.svc.js","modules/overview/overview.ctrl.coffee","modules/overview/overview.ctrl.js","modules/overview/overview.svc.coffee","modules/overview/overview.svc.js","modules/submit/submit.ctrl.coffee","modules/submit/submit.ctrl.js","modules/submit/submit.svc.coffee","modules/submit/submit.svc.js","modules/taskmanager/taskmanager.ctrl.coffee","modules/taskmanager/taskmanager.ctrl.js","modules/taskmanager/taskmanager.svc.coffee","modules/taskmanager/taskmanager.svc.js"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp', ['ui.router', 'angularMoment', 'dndLists'])\r\n\r\n# --------------------------------------\r\n\r\n.run ($rootScope) ->\r\n  $rootScope.sidebarVisible = false\r\n  $rootScope.showSidebar = ->\r\n    $rootScope.sidebarVisible = !$rootScope.sidebarVisible\r\n    $rootScope.sidebarClass = 'force-show'\r\n\r\n# --------------------------------------\r\n\r\n.value 'flinkConfig', {\r\n  jobServer: ''\r\n#  jobServer: 'http://localhost:8081/'\r\n  \"refresh-interval\": 10000\r\n}\r\n\r\n# --------------------------------------\r\n\r\n.run (JobsService, MainService, flinkConfig, $interval) ->\r\n  MainService.loadConfig().then (config) ->\r\n    angular.extend flinkConfig, config\r\n\r\n    JobsService.listJobs()\r\n\r\n    $interval ->\r\n      JobsService.listJobs()\r\n    , flinkConfig[\"refresh-interval\"]\r\n\r\n\r\n# --------------------------------------\r\n\r\n.config ($uiViewScrollProvider) ->\r\n  $uiViewScrollProvider.useAnchorScroll()\r\n\r\n# --------------------------------------\r\n\r\n.run ($rootScope, $state) ->\r\n  $rootScope.$on '$stateChangeStart', (event, toState, toParams, fromState) ->\r\n    if toState.redirectTo\r\n      event.preventDefault()\r\n      $state.go toState.redirectTo, toParams\r\n\r\n# --------------------------------------\r\n\r\n.config ($stateProvider, $urlRouterProvider) ->\r\n  $stateProvider.state \"overview\",\r\n    url: \"/overview\"\r\n    views:\r\n      main:\r\n        templateUrl: \"partials/overview.html\"\r\n        controller: 'OverviewController'\r\n\r\n  .state \"running-jobs\",\r\n    url: \"/running-jobs\"\r\n    views:\r\n      main:\r\n        templateUrl: \"partials/jobs/running-jobs.html\"\r\n        controller: 'RunningJobsController'\r\n\r\n  .state \"completed-jobs\",\r\n    url: \"/completed-jobs\"\r\n    views:\r\n      main:\r\n        templateUrl: \"partials/jobs/completed-jobs.html\"\r\n        controller: 'CompletedJobsController'\r\n\r\n  .state \"single-job\",\r\n    url: \"/jobs/{jobid}\"\r\n    abstract: true\r\n    views:\r\n      main:\r\n        templateUrl: \"partials/jobs/job.html\"\r\n        controller: 'SingleJobController'\r\n\r\n  .state \"single-job.plan\",\r\n    url: \"\"\r\n    redirectTo: \"single-job.plan.subtasks\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/jobs/job.plan.html\"\r\n        controller: 'JobPlanController'\r\n\r\n  .state \"single-job.plan.subtasks\",\r\n    url: \"\"\r\n    views:\r\n      'node-details':\r\n        templateUrl: \"partials/jobs/job.plan.node-list.subtasks.html\"\r\n        controller: 'JobPlanSubtasksController'\r\n\r\n  .state \"single-job.plan.metrics\",\r\n    url: \"/metrics\"\r\n    views:\r\n      'node-details':\r\n        templateUrl: \"partials/jobs/job.plan.node-list.metrics.html\"\r\n        controller: 'JobPlanMetricsController'\r\n\r\n  .state \"single-job.plan.taskmanagers\",\r\n    url: \"/taskmanagers\"\r\n    views:\r\n      'node-details':\r\n        templateUrl: \"partials/jobs/job.plan.node-list.taskmanagers.html\"\r\n        controller: 'JobPlanTaskManagersController'\r\n\r\n  .state \"single-job.plan.accumulators\",\r\n    url: \"/accumulators\"\r\n    views:\r\n      'node-details':\r\n        templateUrl: \"partials/jobs/job.plan.node-list.accumulators.html\"\r\n        controller: 'JobPlanAccumulatorsController'\r\n\r\n  .state \"single-job.plan.checkpoints\",\r\n    url: \"/checkpoints\"\r\n    views:\r\n      'node-details':\r\n        templateUrl: \"partials/jobs/job.plan.node-list.checkpoints.html\"\r\n        controller: 'JobPlanCheckpointsController'\r\n\r\n  .state \"single-job.plan.backpressure\",\r\n    url: \"/backpressure\"\r\n    views:\r\n      'node-details':\r\n        templateUrl: \"partials/jobs/job.plan.node-list.backpressure.html\"\r\n        controller: 'JobPlanBackPressureController'\r\n\r\n  .state \"single-job.timeline\",\r\n    url: \"/timeline\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/jobs/job.timeline.html\"\r\n\r\n  .state \"single-job.timeline.vertex\",\r\n    url: \"/{vertexId}\"\r\n    views:\r\n      vertex:\r\n        templateUrl: \"partials/jobs/job.timeline.vertex.html\"\r\n        controller: 'JobTimelineVertexController'\r\n\r\n  .state \"single-job.exceptions\",\r\n    url: \"/exceptions\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/jobs/job.exceptions.html\"\r\n        controller: 'JobExceptionsController'\r\n\r\n  .state \"single-job.config\",\r\n    url: \"/config\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/jobs/job.config.html\"\r\n\r\n  .state \"all-manager\",\r\n    url: \"/taskmanagers\"\r\n    views:\r\n      main:\r\n        templateUrl: \"partials/taskmanager/index.html\"\r\n        controller: 'AllTaskManagersController'\r\n\r\n  .state \"single-manager\",\r\n      url: \"/taskmanager/{taskmanagerid}\"\r\n      views:\r\n        main:\r\n          templateUrl: \"partials/taskmanager/taskmanager.html\"\r\n\r\n  .state \"single-manager.metrics\",\r\n    url: \"/metrics\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/taskmanager/taskmanager.metrics.html\"\r\n        controller: 'SingleTaskManagerController'\r\n\r\n  .state \"single-manager.stdout\",\r\n    url: \"/stdout\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/taskmanager/taskmanager.stdout.html\"\r\n        controller: 'SingleTaskManagerStdoutController'\r\n\r\n  .state \"single-manager.log\",\r\n    url: \"/log\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/taskmanager/taskmanager.log.html\"\r\n        controller: 'SingleTaskManagerLogsController'\r\n\r\n  .state \"jobmanager\",\r\n      url: \"/jobmanager\"\r\n      views:\r\n        main:\r\n          templateUrl: \"partials/jobmanager/index.html\"\r\n\r\n  .state \"jobmanager.config\",\r\n    url: \"/config\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/jobmanager/config.html\"\r\n        controller: 'JobManagerConfigController'\r\n\r\n  .state \"jobmanager.stdout\",\r\n    url: \"/stdout\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/jobmanager/stdout.html\"\r\n        controller: 'JobManagerStdoutController'\r\n\r\n  .state \"jobmanager.log\",\r\n    url: \"/log\"\r\n    views:\r\n      details:\r\n        templateUrl: \"partials/jobmanager/log.html\"\r\n        controller: 'JobManagerLogsController'\r\n\r\n  .state \"submit\",\r\n      url: \"/submit\"\r\n      views:\r\n        main:\r\n          templateUrl: \"partials/submit.html\"\r\n          controller: \"JobSubmitController\"\r\n\r\n  $urlRouterProvider.otherwise \"/overview\"\r\n","angular.module('flinkApp', ['ui.router', 'angularMoment', 'dndLists']).run(function($rootScope) {\n  $rootScope.sidebarVisible = false;\n  return $rootScope.showSidebar = function() {\n    $rootScope.sidebarVisible = !$rootScope.sidebarVisible;\n    return $rootScope.sidebarClass = 'force-show';\n  };\n}).value('flinkConfig', {\n  jobServer: '',\n  \"refresh-interval\": 10000\n}).run(function(JobsService, MainService, flinkConfig, $interval) {\n  return MainService.loadConfig().then(function(config) {\n    angular.extend(flinkConfig, config);\n    JobsService.listJobs();\n    return $interval(function() {\n      return JobsService.listJobs();\n    }, flinkConfig[\"refresh-interval\"]);\n  });\n}).config(function($uiViewScrollProvider) {\n  return $uiViewScrollProvider.useAnchorScroll();\n}).run(function($rootScope, $state) {\n  return $rootScope.$on('$stateChangeStart', function(event, toState, toParams, fromState) {\n    if (toState.redirectTo) {\n      event.preventDefault();\n      return $state.go(toState.redirectTo, toParams);\n    }\n  });\n}).config(function($stateProvider, $urlRouterProvider) {\n  $stateProvider.state(\"overview\", {\n    url: \"/overview\",\n    views: {\n      main: {\n        templateUrl: \"partials/overview.html\",\n        controller: 'OverviewController'\n      }\n    }\n  }).state(\"running-jobs\", {\n    url: \"/running-jobs\",\n    views: {\n      main: {\n        templateUrl: \"partials/jobs/running-jobs.html\",\n        controller: 'RunningJobsController'\n      }\n    }\n  }).state(\"completed-jobs\", {\n    url: \"/completed-jobs\",\n    views: {\n      main: {\n        templateUrl: \"partials/jobs/completed-jobs.html\",\n        controller: 'CompletedJobsController'\n      }\n    }\n  }).state(\"single-job\", {\n    url: \"/jobs/{jobid}\",\n    abstract: true,\n    views: {\n      main: {\n        templateUrl: \"partials/jobs/job.html\",\n        controller: 'SingleJobController'\n      }\n    }\n  }).state(\"single-job.plan\", {\n    url: \"\",\n    redirectTo: \"single-job.plan.subtasks\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.plan.html\",\n        controller: 'JobPlanController'\n      }\n    }\n  }).state(\"single-job.plan.subtasks\", {\n    url: \"\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.subtasks.html\",\n        controller: 'JobPlanSubtasksController'\n      }\n    }\n  }).state(\"single-job.plan.metrics\", {\n    url: \"/metrics\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.metrics.html\",\n        controller: 'JobPlanMetricsController'\n      }\n    }\n  }).state(\"single-job.plan.taskmanagers\", {\n    url: \"/taskmanagers\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.taskmanagers.html\",\n        controller: 'JobPlanTaskManagersController'\n      }\n    }\n  }).state(\"single-job.plan.accumulators\", {\n    url: \"/accumulators\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.accumulators.html\",\n        controller: 'JobPlanAccumulatorsController'\n      }\n    }\n  }).state(\"single-job.plan.checkpoints\", {\n    url: \"/checkpoints\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.checkpoints.html\",\n        controller: 'JobPlanCheckpointsController'\n      }\n    }\n  }).state(\"single-job.plan.backpressure\", {\n    url: \"/backpressure\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.backpressure.html\",\n        controller: 'JobPlanBackPressureController'\n      }\n    }\n  }).state(\"single-job.timeline\", {\n    url: \"/timeline\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.timeline.html\"\n      }\n    }\n  }).state(\"single-job.timeline.vertex\", {\n    url: \"/{vertexId}\",\n    views: {\n      vertex: {\n        templateUrl: \"partials/jobs/job.timeline.vertex.html\",\n        controller: 'JobTimelineVertexController'\n      }\n    }\n  }).state(\"single-job.exceptions\", {\n    url: \"/exceptions\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.exceptions.html\",\n        controller: 'JobExceptionsController'\n      }\n    }\n  }).state(\"single-job.config\", {\n    url: \"/config\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.config.html\"\n      }\n    }\n  }).state(\"all-manager\", {\n    url: \"/taskmanagers\",\n    views: {\n      main: {\n        templateUrl: \"partials/taskmanager/index.html\",\n        controller: 'AllTaskManagersController'\n      }\n    }\n  }).state(\"single-manager\", {\n    url: \"/taskmanager/{taskmanagerid}\",\n    views: {\n      main: {\n        templateUrl: \"partials/taskmanager/taskmanager.html\"\n      }\n    }\n  }).state(\"single-manager.metrics\", {\n    url: \"/metrics\",\n    views: {\n      details: {\n        templateUrl: \"partials/taskmanager/taskmanager.metrics.html\",\n        controller: 'SingleTaskManagerController'\n      }\n    }\n  }).state(\"single-manager.stdout\", {\n    url: \"/stdout\",\n    views: {\n      details: {\n        templateUrl: \"partials/taskmanager/taskmanager.stdout.html\",\n        controller: 'SingleTaskManagerStdoutController'\n      }\n    }\n  }).state(\"single-manager.log\", {\n    url: \"/log\",\n    views: {\n      details: {\n        templateUrl: \"partials/taskmanager/taskmanager.log.html\",\n        controller: 'SingleTaskManagerLogsController'\n      }\n    }\n  }).state(\"jobmanager\", {\n    url: \"/jobmanager\",\n    views: {\n      main: {\n        templateUrl: \"partials/jobmanager/index.html\"\n      }\n    }\n  }).state(\"jobmanager.config\", {\n    url: \"/config\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobmanager/config.html\",\n        controller: 'JobManagerConfigController'\n      }\n    }\n  }).state(\"jobmanager.stdout\", {\n    url: \"/stdout\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobmanager/stdout.html\",\n        controller: 'JobManagerStdoutController'\n      }\n    }\n  }).state(\"jobmanager.log\", {\n    url: \"/log\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobmanager/log.html\",\n        controller: 'JobManagerLogsController'\n      }\n    }\n  }).state(\"submit\", {\n    url: \"/submit\",\n    views: {\n      main: {\n        templateUrl: \"partials/submit.html\",\n        controller: \"JobSubmitController\"\n      }\n    }\n  });\n  return $urlRouterProvider.otherwise(\"/overview\");\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n# ----------------------------------------------\r\n\r\n.directive 'bsLabel', (JobsService) ->\r\n  transclude: true\r\n  replace: true\r\n  scope: \r\n    getLabelClass: \"&\"\r\n    status: \"@\"\r\n\r\n  template: \"<span title='{{status}}' ng-class='getLabelClass()'><ng-transclude></ng-transclude></span>\"\r\n  \r\n  link: (scope, element, attrs) ->\r\n    scope.getLabelClass = ->\r\n      'label label-' + JobsService.translateLabelState(attrs.status)\r\n\r\n# ----------------------------------------------\r\n\r\n.directive 'bpLabel', (JobsService) ->\r\n  transclude: true\r\n  replace: true\r\n  scope:\r\n    getBackPressureLabelClass: \"&\"\r\n    status: \"@\"\r\n\r\n  template: \"<span title='{{status}}' ng-class='getBackPressureLabelClass()'><ng-transclude></ng-transclude></span>\"\r\n\r\n  link: (scope, element, attrs) ->\r\n    scope.getBackPressureLabelClass = ->\r\n      'label label-' + JobsService.translateBackPressureLabelState(attrs.status)\r\n\r\n# ----------------------------------------------\r\n\r\n.directive 'indicatorPrimary', (JobsService) ->\r\n  replace: true\r\n  scope: \r\n    getLabelClass: \"&\"\r\n    status: '@'\r\n\r\n  template: \"<i title='{{status}}' ng-class='getLabelClass()' />\"\r\n  \r\n  link: (scope, element, attrs) ->\r\n    scope.getLabelClass = ->\r\n      'fa fa-circle indicator indicator-' + JobsService.translateLabelState(attrs.status)\r\n\r\n# ----------------------------------------------\r\n\r\n.directive 'tableProperty', ->\r\n  replace: true\r\n  scope:\r\n    value: '='\r\n\r\n  template: \"<td title=\\\"{{value || 'None'}}\\\">{{value || 'None'}}</td>\"\r\n","angular.module('flinkApp').directive('bsLabel', function(JobsService) {\n  return {\n    transclude: true,\n    replace: true,\n    scope: {\n      getLabelClass: \"&\",\n      status: \"@\"\n    },\n    template: \"<span title='{{status}}' ng-class='getLabelClass()'><ng-transclude></ng-transclude></span>\",\n    link: function(scope, element, attrs) {\n      return scope.getLabelClass = function() {\n        return 'label label-' + JobsService.translateLabelState(attrs.status);\n      };\n    }\n  };\n}).directive('bpLabel', function(JobsService) {\n  return {\n    transclude: true,\n    replace: true,\n    scope: {\n      getBackPressureLabelClass: \"&\",\n      status: \"@\"\n    },\n    template: \"<span title='{{status}}' ng-class='getBackPressureLabelClass()'><ng-transclude></ng-transclude></span>\",\n    link: function(scope, element, attrs) {\n      return scope.getBackPressureLabelClass = function() {\n        return 'label label-' + JobsService.translateBackPressureLabelState(attrs.status);\n      };\n    }\n  };\n}).directive('indicatorPrimary', function(JobsService) {\n  return {\n    replace: true,\n    scope: {\n      getLabelClass: \"&\",\n      status: '@'\n    },\n    template: \"<i title='{{status}}' ng-class='getLabelClass()' />\",\n    link: function(scope, element, attrs) {\n      return scope.getLabelClass = function() {\n        return 'fa fa-circle indicator indicator-' + JobsService.translateLabelState(attrs.status);\n      };\n    }\n  };\n}).directive('tableProperty', function() {\n  return {\n    replace: true,\n    scope: {\n      value: '='\n    },\n    template: \"<td title=\\\"{{value || 'None'}}\\\">{{value || 'None'}}</td>\"\n  };\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.filter \"amDurationFormatExtended\", (angularMomentConfig) ->\r\n  amDurationFormatExtendedFilter = (value, format, durationFormat) ->\r\n    return \"\"  if typeof value is \"undefined\" or value is null\r\n\r\n    moment.duration(value, format).format(durationFormat, { trim: false })\r\n\r\n  amDurationFormatExtendedFilter.$stateful = angularMomentConfig.statefulFilters\r\n\r\n  amDurationFormatExtendedFilter\r\n\r\n.filter \"humanizeDuration\", ->\r\n  (value, short) ->\r\n    return \"\" if typeof value is \"undefined\" or value is null\r\n    ms = value % 1000\r\n    x = Math.floor(value / 1000)\r\n    seconds = x % 60\r\n    x = Math.floor(x / 60)\r\n    minutes = x % 60\r\n    x = Math.floor(x / 60)\r\n    hours = x % 24\r\n    x = Math.floor(x / 24)\r\n    days = x\r\n    if days == 0\r\n      if hours == 0\r\n        if minutes == 0\r\n          if seconds == 0\r\n            return ms + \"ms\"\r\n          else\r\n            return seconds + \"s \"\r\n        else\r\n          return minutes + \"m \" + seconds + \"s\"\r\n      else\r\n        if short then return hours + \"h \" + minutes + \"m\" else return hours + \"h \" + minutes + \"m \" + seconds + \"s\"\r\n    else\r\n      if short then return days + \"d \" + hours + \"h\" else return days + \"d \" + hours + \"h \" + minutes + \"m \" + seconds + \"s\"\r\n\r\n.filter \"humanizeText\", ->\r\n  (text) ->\r\n    # TODO: extend... a lot\r\n    if text then text.replace(/&gt;/g, \">\").replace(/<br\\/>/g,\"\") else ''\r\n\r\n.filter \"humanizeBytes\", ->\r\n  (bytes) ->\r\n    units = [\"B\", \"KB\", \"MB\", \"GB\", \"TB\", \"PB\", \"EB\"]\r\n    converter = (value, power) ->\r\n      base = Math.pow(1024, power)\r\n      if value < base\r\n        return (value / base).toFixed(2) + \" \" + units[power]\r\n      else if value < base * 1000\r\n        return (value / base).toPrecision(3) + \" \" + units[power]\r\n      else\r\n        return converter(value, power + 1)\r\n    return \"\" if typeof bytes is \"undefined\" or bytes is null\r\n    if bytes < 1000 then bytes + \" B\" else converter(bytes, 1)\r\n\r\n.filter \"toLocaleString\", ->\r\n  (text) -> text.toLocaleString()\r\n\r\n.filter \"toUpperCase\", ->\r\n  (text) -> text.toUpperCase()\r\n","angular.module('flinkApp').filter(\"amDurationFormatExtended\", function(angularMomentConfig) {\n  var amDurationFormatExtendedFilter;\n  amDurationFormatExtendedFilter = function(value, format, durationFormat) {\n    if (typeof value === \"undefined\" || value === null) {\n      return \"\";\n    }\n    return moment.duration(value, format).format(durationFormat, {\n      trim: false\n    });\n  };\n  amDurationFormatExtendedFilter.$stateful = angularMomentConfig.statefulFilters;\n  return amDurationFormatExtendedFilter;\n}).filter(\"humanizeDuration\", function() {\n  return function(value, short) {\n    var days, hours, minutes, ms, seconds, x;\n    if (typeof value === \"undefined\" || value === null) {\n      return \"\";\n    }\n    ms = value % 1000;\n    x = Math.floor(value / 1000);\n    seconds = x % 60;\n    x = Math.floor(x / 60);\n    minutes = x % 60;\n    x = Math.floor(x / 60);\n    hours = x % 24;\n    x = Math.floor(x / 24);\n    days = x;\n    if (days === 0) {\n      if (hours === 0) {\n        if (minutes === 0) {\n          if (seconds === 0) {\n            return ms + \"ms\";\n          } else {\n            return seconds + \"s \";\n          }\n        } else {\n          return minutes + \"m \" + seconds + \"s\";\n        }\n      } else {\n        if (short) {\n          return hours + \"h \" + minutes + \"m\";\n        } else {\n          return hours + \"h \" + minutes + \"m \" + seconds + \"s\";\n        }\n      }\n    } else {\n      if (short) {\n        return days + \"d \" + hours + \"h\";\n      } else {\n        return days + \"d \" + hours + \"h \" + minutes + \"m \" + seconds + \"s\";\n      }\n    }\n  };\n}).filter(\"humanizeText\", function() {\n  return function(text) {\n    if (text) {\n      return text.replace(/&gt;/g, \">\").replace(/<br\\/>/g, \"\");\n    } else {\n      return '';\n    }\n  };\n}).filter(\"humanizeBytes\", function() {\n  return function(bytes) {\n    var converter, units;\n    units = [\"B\", \"KB\", \"MB\", \"GB\", \"TB\", \"PB\", \"EB\"];\n    converter = function(value, power) {\n      var base;\n      base = Math.pow(1024, power);\n      if (value < base) {\n        return (value / base).toFixed(2) + \" \" + units[power];\n      } else if (value < base * 1000) {\n        return (value / base).toPrecision(3) + \" \" + units[power];\n      } else {\n        return converter(value, power + 1);\n      }\n    };\n    if (typeof bytes === \"undefined\" || bytes === null) {\n      return \"\";\n    }\n    if (bytes < 1000) {\n      return bytes + \" B\";\n    } else {\n      return converter(bytes, 1);\n    }\n  };\n}).filter(\"toLocaleString\", function() {\n  return function(text) {\n    return text.toLocaleString();\n  };\n}).filter(\"toUpperCase\", function() {\n  return function(text) {\n    return text.toUpperCase();\n  };\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.service 'MainService', ($http, flinkConfig, $q) ->\r\n  @loadConfig = ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get flinkConfig.jobServer + \"config\"\r\n    .success (data, status, headers, config) ->\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n\r\n  @\r\n","angular.module('flinkApp').service('MainService', function($http, flinkConfig, $q) {\n  this.loadConfig = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"config\").success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.controller 'JobManagerConfigController', ($scope, JobManagerConfigService) ->\r\n  JobManagerConfigService.loadConfig().then (data) ->\r\n    if !$scope.jobmanager?\r\n      $scope.jobmanager = {}\r\n    $scope.jobmanager['config'] = data\r\n\r\n.controller 'JobManagerLogsController', ($scope, JobManagerLogsService) ->\r\n  JobManagerLogsService.loadLogs().then (data) ->\r\n    if !$scope.jobmanager?\r\n      $scope.jobmanager = {}\r\n    $scope.jobmanager['log'] = data\r\n\r\n  $scope.reloadData = () ->\r\n    JobManagerLogsService.loadLogs().then (data) ->\r\n      $scope.jobmanager['log'] = data\r\n\r\n.controller 'JobManagerStdoutController', ($scope, JobManagerStdoutService) ->\r\n  JobManagerStdoutService.loadStdout().then (data) ->\r\n    if !$scope.jobmanager?\r\n      $scope.jobmanager = {}\r\n    $scope.jobmanager['stdout'] = data\r\n\r\n  $scope.reloadData = () ->\r\n    JobManagerStdoutService.loadStdout().then (data) ->\r\n      $scope.jobmanager['stdout'] = data\r\n","angular.module('flinkApp').controller('JobManagerConfigController', function($scope, JobManagerConfigService) {\n  return JobManagerConfigService.loadConfig().then(function(data) {\n    if ($scope.jobmanager == null) {\n      $scope.jobmanager = {};\n    }\n    return $scope.jobmanager['config'] = data;\n  });\n}).controller('JobManagerLogsController', function($scope, JobManagerLogsService) {\n  JobManagerLogsService.loadLogs().then(function(data) {\n    if ($scope.jobmanager == null) {\n      $scope.jobmanager = {};\n    }\n    return $scope.jobmanager['log'] = data;\n  });\n  return $scope.reloadData = function() {\n    return JobManagerLogsService.loadLogs().then(function(data) {\n      return $scope.jobmanager['log'] = data;\n    });\n  };\n}).controller('JobManagerStdoutController', function($scope, JobManagerStdoutService) {\n  JobManagerStdoutService.loadStdout().then(function(data) {\n    if ($scope.jobmanager == null) {\n      $scope.jobmanager = {};\n    }\n    return $scope.jobmanager['stdout'] = data;\n  });\n  return $scope.reloadData = function() {\n    return JobManagerStdoutService.loadStdout().then(function(data) {\n      return $scope.jobmanager['stdout'] = data;\n    });\n  };\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.service 'JobManagerConfigService', ($http, flinkConfig, $q) ->\r\n  config = {}\r\n\r\n  @loadConfig = ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"jobmanager/config\")\r\n    .success (data, status, headers, config) ->\r\n      config = data\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @\r\n\r\n.service 'JobManagerLogsService', ($http, flinkConfig, $q) ->\r\n  logs = {}\r\n\r\n  @loadLogs = ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"jobmanager/log\")\r\n    .success (data, status, headers, config) ->\r\n      logs = data\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @\r\n\r\n.service 'JobManagerStdoutService', ($http, flinkConfig, $q) ->\r\n  stdout = {}\r\n\r\n  @loadStdout = ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"jobmanager/stdout\")\r\n    .success (data, status, headers, config) ->\r\n      stdout = data\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @\r\n","angular.module('flinkApp').service('JobManagerConfigService', function($http, flinkConfig, $q) {\n  var config;\n  config = {};\n  this.loadConfig = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobmanager/config\").success(function(data, status, headers, config) {\n      config = data;\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n}).service('JobManagerLogsService', function($http, flinkConfig, $q) {\n  var logs;\n  logs = {};\n  this.loadLogs = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobmanager/log\").success(function(data, status, headers, config) {\n      logs = data;\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n}).service('JobManagerStdoutService', function($http, flinkConfig, $q) {\n  var stdout;\n  stdout = {};\n  this.loadStdout = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobmanager/stdout\").success(function(data, status, headers, config) {\n      stdout = data;\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.controller 'RunningJobsController', ($scope, $state, $stateParams, JobsService) ->\r\n  $scope.jobObserver = ->\r\n    $scope.jobs = JobsService.getJobs('running')\r\n\r\n  JobsService.registerObserver($scope.jobObserver)\r\n  $scope.$on '$destroy', ->\r\n    JobsService.unRegisterObserver($scope.jobObserver)\r\n\r\n  $scope.jobObserver()\r\n\r\n# --------------------------------------\r\n\r\n.controller 'CompletedJobsController', ($scope, $state, $stateParams, JobsService) ->\r\n  $scope.jobObserver = ->\r\n    $scope.jobs = JobsService.getJobs('finished')\r\n\r\n  JobsService.registerObserver($scope.jobObserver)\r\n  $scope.$on '$destroy', ->\r\n    JobsService.unRegisterObserver($scope.jobObserver)\r\n\r\n  $scope.jobObserver()\r\n\r\n# --------------------------------------\r\n\r\n.controller 'SingleJobController', ($scope, $state, $stateParams, JobsService, MetricsService, $rootScope, flinkConfig, $interval) ->\r\n  $scope.jobid = $stateParams.jobid\r\n  $scope.job = null\r\n  $scope.plan = null\r\n  $scope.vertices = null\r\n  $scope.jobCheckpointStats = null\r\n  $scope.showHistory = false\r\n  $scope.backPressureOperatorStats = {}\r\n\r\n  JobsService.loadJob($stateParams.jobid).then (data) ->\r\n    $scope.job = data\r\n    $scope.plan = data.plan\r\n    $scope.vertices = data.vertices\r\n    MetricsService.setupMetrics($stateParams.jobid, data.vertices)\r\n\r\n  refresher = $interval ->\r\n    JobsService.loadJob($stateParams.jobid).then (data) ->\r\n      $scope.job = data\r\n\r\n      $scope.$broadcast 'reload'\r\n\r\n  , flinkConfig[\"refresh-interval\"]\r\n\r\n  $scope.$on '$destroy', ->\r\n    $scope.job = null\r\n    $scope.plan = null\r\n    $scope.vertices = null\r\n    $scope.jobCheckpointStats = null\r\n    $scope.backPressureOperatorStats = null\r\n\r\n    $interval.cancel(refresher)\r\n\r\n  $scope.cancelJob = (cancelEvent) ->\r\n    angular.element(cancelEvent.currentTarget).removeClass(\"btn\").removeClass(\"btn-default\").html('Cancelling...')\r\n    JobsService.cancelJob($stateParams.jobid).then (data) ->\r\n      {}\r\n\r\n  $scope.stopJob = (stopEvent) ->\r\n    angular.element(stopEvent.currentTarget).removeClass(\"btn\").removeClass(\"btn-default\").html('Stopping...')\r\n    JobsService.stopJob($stateParams.jobid).then (data) ->\r\n      {}\r\n\r\n  $scope.toggleHistory = ->\r\n    $scope.showHistory = !$scope.showHistory\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobPlanController', ($scope, $state, $stateParams, $window, JobsService) ->\r\n  $scope.nodeid = null\r\n  $scope.nodeUnfolded = false\r\n  $scope.stateList = JobsService.stateList()\r\n\r\n  $scope.changeNode = (nodeid) ->\r\n    if nodeid != $scope.nodeid\r\n      $scope.nodeid = nodeid\r\n      $scope.vertex = null\r\n      $scope.subtasks = null\r\n      $scope.accumulators = null\r\n      $scope.operatorCheckpointStats = null\r\n\r\n      $scope.$broadcast 'reload'\r\n      $scope.$broadcast 'node:change', $scope.nodeid\r\n\r\n    else\r\n      $scope.nodeid = null\r\n      $scope.nodeUnfolded = false\r\n      $scope.vertex = null\r\n      $scope.subtasks = null\r\n      $scope.accumulators = null\r\n      $scope.operatorCheckpointStats = null\r\n\r\n  $scope.deactivateNode = ->\r\n    $scope.nodeid = null\r\n    $scope.nodeUnfolded = false\r\n    $scope.vertex = null\r\n    $scope.subtasks = null\r\n    $scope.accumulators = null\r\n    $scope.operatorCheckpointStats = null\r\n\r\n  $scope.toggleFold = ->\r\n    $scope.nodeUnfolded = !$scope.nodeUnfolded\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobPlanSubtasksController', ($scope, JobsService) ->\r\n  getSubtasks = ->\r\n    JobsService.getSubtasks($scope.nodeid).then (data) ->\r\n      $scope.subtasks = data\r\n\r\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.st)\r\n    getSubtasks()\r\n\r\n  $scope.$on 'reload', (event) ->\r\n    getSubtasks() if $scope.nodeid\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobPlanTaskManagersController', ($scope, JobsService) ->\r\n  getTaskManagers = ->\r\n    JobsService.getTaskManagers($scope.nodeid).then (data) ->\r\n      $scope.taskmanagers = data\r\n\r\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.st)\r\n    getTaskManagers()\r\n\r\n  $scope.$on 'reload', (event) ->\r\n    getTaskManagers() if $scope.nodeid\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobPlanAccumulatorsController', ($scope, JobsService) ->\r\n  getAccumulators = ->\r\n    JobsService.getAccumulators($scope.nodeid).then (data) ->\r\n      $scope.accumulators = data.main\r\n      $scope.subtaskAccumulators = data.subtasks\r\n\r\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.accumulators)\r\n    getAccumulators()\r\n\r\n  $scope.$on 'reload', (event) ->\r\n    getAccumulators() if $scope.nodeid\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobPlanCheckpointsController', ($scope, JobsService) ->\r\n  getJobCheckpointStats = ->\r\n    JobsService.getJobCheckpointStats($scope.jobid).then (data) ->\r\n      $scope.jobCheckpointStats = data\r\n\r\n  getOperatorCheckpointStats = ->\r\n    JobsService.getOperatorCheckpointStats($scope.nodeid).then (data) ->\r\n      $scope.operatorCheckpointStats = data.operatorStats\r\n      $scope.subtasksCheckpointStats = data.subtasksStats\r\n\r\n  # Get the per job stats\r\n  getJobCheckpointStats()\r\n\r\n  # Get the per operator stats\r\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.operatorCheckpointStats)\r\n    getOperatorCheckpointStats()\r\n\r\n  $scope.$on 'reload', (event) ->\r\n    getJobCheckpointStats()\r\n    getOperatorCheckpointStats() if $scope.nodeid\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobPlanBackPressureController', ($scope, JobsService) ->\r\n  getOperatorBackPressure = ->\r\n    $scope.now = Date.now()\r\n\r\n    if $scope.nodeid\r\n      JobsService.getOperatorBackPressure($scope.nodeid).then (data) ->\r\n        $scope.backPressureOperatorStats[$scope.nodeid] = data\r\n\r\n  getOperatorBackPressure()\r\n\r\n  $scope.$on 'reload', (event) ->\r\n    getOperatorBackPressure()\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobTimelineVertexController', ($scope, $state, $stateParams, JobsService) ->\r\n  getVertex = ->\r\n    JobsService.getVertex($stateParams.vertexId).then (data) ->\r\n      $scope.vertex = data\r\n\r\n  getVertex()\r\n\r\n  $scope.$on 'reload', (event) ->\r\n    getVertex()\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobExceptionsController', ($scope, $state, $stateParams, JobsService) ->\r\n  JobsService.loadExceptions().then (data) ->\r\n    $scope.exceptions = data\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobPropertiesController', ($scope, JobsService) ->\r\n  $scope.changeNode = (nodeid) ->\r\n    if nodeid != $scope.nodeid\r\n      $scope.nodeid = nodeid\r\n\r\n      JobsService.getNode(nodeid).then (data) ->\r\n        $scope.node = data\r\n\r\n    else\r\n      $scope.nodeid = null\r\n      $scope.node = null\r\n\r\n# --------------------------------------\r\n\r\n.controller 'JobPlanMetricsController', ($scope, JobsService, MetricsService) ->\r\n  $scope.dragging = false\r\n  $scope.window = MetricsService.getWindow()\r\n  $scope.availableMetrics = null\r\n\r\n  $scope.$on '$destroy', ->\r\n    MetricsService.unRegisterObserver()\r\n\r\n  loadMetrics = ->\r\n    JobsService.getVertex($scope.nodeid).then (data) ->\r\n      $scope.vertex = data\r\n\r\n    MetricsService.getAvailableMetrics($scope.jobid, $scope.nodeid).then (data) ->\r\n      $scope.availableMetrics = data\r\n      $scope.metrics = MetricsService.getMetricsSetup($scope.jobid, $scope.nodeid).names\r\n\r\n      MetricsService.registerObserver($scope.jobid, $scope.nodeid, (data) ->\r\n        $scope.$broadcast \"metrics:data:update\", data.timestamp, data.values\r\n      )\r\n\r\n  $scope.dropped = (event, index, item, external, type) ->\r\n\r\n    MetricsService.orderMetrics($scope.jobid, $scope.nodeid, item, index)\r\n    $scope.$broadcast \"metrics:refresh\", item\r\n    loadMetrics()\r\n    false\r\n\r\n  $scope.dragStart = ->\r\n    $scope.dragging = true\r\n\r\n  $scope.dragEnd = ->\r\n    $scope.dragging = false\r\n\r\n  $scope.addMetric = (metric) ->\r\n    MetricsService.addMetric($scope.jobid, $scope.nodeid, metric.id)\r\n    loadMetrics()\r\n\r\n  $scope.removeMetric = (metric) ->\r\n    MetricsService.removeMetric($scope.jobid, $scope.nodeid, metric)\r\n    loadMetrics()\r\n\r\n  $scope.setMetricSize = (metric, size) ->\r\n    MetricsService.setMetricSize($scope.jobid, $scope.nodeid, metric, size)\r\n    loadMetrics()\r\n\r\n  $scope.getValues = (metric) ->\r\n    MetricsService.getValues($scope.jobid, $scope.nodeid, metric)\r\n\r\n  $scope.$on 'node:change', (event, nodeid) ->\r\n    loadMetrics() if !$scope.dragging\r\n\r\n  loadMetrics() if $scope.nodeid\r\n\r\n# --------------------------------------\r\n","angular.module('flinkApp').controller('RunningJobsController', function($scope, $state, $stateParams, JobsService) {\n  $scope.jobObserver = function() {\n    return $scope.jobs = JobsService.getJobs('running');\n  };\n  JobsService.registerObserver($scope.jobObserver);\n  $scope.$on('$destroy', function() {\n    return JobsService.unRegisterObserver($scope.jobObserver);\n  });\n  return $scope.jobObserver();\n}).controller('CompletedJobsController', function($scope, $state, $stateParams, JobsService) {\n  $scope.jobObserver = function() {\n    return $scope.jobs = JobsService.getJobs('finished');\n  };\n  JobsService.registerObserver($scope.jobObserver);\n  $scope.$on('$destroy', function() {\n    return JobsService.unRegisterObserver($scope.jobObserver);\n  });\n  return $scope.jobObserver();\n}).controller('SingleJobController', function($scope, $state, $stateParams, JobsService, MetricsService, $rootScope, flinkConfig, $interval) {\n  var refresher;\n  $scope.jobid = $stateParams.jobid;\n  $scope.job = null;\n  $scope.plan = null;\n  $scope.vertices = null;\n  $scope.jobCheckpointStats = null;\n  $scope.showHistory = false;\n  $scope.backPressureOperatorStats = {};\n  JobsService.loadJob($stateParams.jobid).then(function(data) {\n    $scope.job = data;\n    $scope.plan = data.plan;\n    $scope.vertices = data.vertices;\n    return MetricsService.setupMetrics($stateParams.jobid, data.vertices);\n  });\n  refresher = $interval(function() {\n    return JobsService.loadJob($stateParams.jobid).then(function(data) {\n      $scope.job = data;\n      return $scope.$broadcast('reload');\n    });\n  }, flinkConfig[\"refresh-interval\"]);\n  $scope.$on('$destroy', function() {\n    $scope.job = null;\n    $scope.plan = null;\n    $scope.vertices = null;\n    $scope.jobCheckpointStats = null;\n    $scope.backPressureOperatorStats = null;\n    return $interval.cancel(refresher);\n  });\n  $scope.cancelJob = function(cancelEvent) {\n    angular.element(cancelEvent.currentTarget).removeClass(\"btn\").removeClass(\"btn-default\").html('Cancelling...');\n    return JobsService.cancelJob($stateParams.jobid).then(function(data) {\n      return {};\n    });\n  };\n  $scope.stopJob = function(stopEvent) {\n    angular.element(stopEvent.currentTarget).removeClass(\"btn\").removeClass(\"btn-default\").html('Stopping...');\n    return JobsService.stopJob($stateParams.jobid).then(function(data) {\n      return {};\n    });\n  };\n  return $scope.toggleHistory = function() {\n    return $scope.showHistory = !$scope.showHistory;\n  };\n}).controller('JobPlanController', function($scope, $state, $stateParams, $window, JobsService) {\n  $scope.nodeid = null;\n  $scope.nodeUnfolded = false;\n  $scope.stateList = JobsService.stateList();\n  $scope.changeNode = function(nodeid) {\n    if (nodeid !== $scope.nodeid) {\n      $scope.nodeid = nodeid;\n      $scope.vertex = null;\n      $scope.subtasks = null;\n      $scope.accumulators = null;\n      $scope.operatorCheckpointStats = null;\n      $scope.$broadcast('reload');\n      return $scope.$broadcast('node:change', $scope.nodeid);\n    } else {\n      $scope.nodeid = null;\n      $scope.nodeUnfolded = false;\n      $scope.vertex = null;\n      $scope.subtasks = null;\n      $scope.accumulators = null;\n      return $scope.operatorCheckpointStats = null;\n    }\n  };\n  $scope.deactivateNode = function() {\n    $scope.nodeid = null;\n    $scope.nodeUnfolded = false;\n    $scope.vertex = null;\n    $scope.subtasks = null;\n    $scope.accumulators = null;\n    return $scope.operatorCheckpointStats = null;\n  };\n  return $scope.toggleFold = function() {\n    return $scope.nodeUnfolded = !$scope.nodeUnfolded;\n  };\n}).controller('JobPlanSubtasksController', function($scope, JobsService) {\n  var getSubtasks;\n  getSubtasks = function() {\n    return JobsService.getSubtasks($scope.nodeid).then(function(data) {\n      return $scope.subtasks = data;\n    });\n  };\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.st)) {\n    getSubtasks();\n  }\n  return $scope.$on('reload', function(event) {\n    if ($scope.nodeid) {\n      return getSubtasks();\n    }\n  });\n}).controller('JobPlanTaskManagersController', function($scope, JobsService) {\n  var getTaskManagers;\n  getTaskManagers = function() {\n    return JobsService.getTaskManagers($scope.nodeid).then(function(data) {\n      return $scope.taskmanagers = data;\n    });\n  };\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.st)) {\n    getTaskManagers();\n  }\n  return $scope.$on('reload', function(event) {\n    if ($scope.nodeid) {\n      return getTaskManagers();\n    }\n  });\n}).controller('JobPlanAccumulatorsController', function($scope, JobsService) {\n  var getAccumulators;\n  getAccumulators = function() {\n    return JobsService.getAccumulators($scope.nodeid).then(function(data) {\n      $scope.accumulators = data.main;\n      return $scope.subtaskAccumulators = data.subtasks;\n    });\n  };\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.accumulators)) {\n    getAccumulators();\n  }\n  return $scope.$on('reload', function(event) {\n    if ($scope.nodeid) {\n      return getAccumulators();\n    }\n  });\n}).controller('JobPlanCheckpointsController', function($scope, JobsService) {\n  var getJobCheckpointStats, getOperatorCheckpointStats;\n  getJobCheckpointStats = function() {\n    return JobsService.getJobCheckpointStats($scope.jobid).then(function(data) {\n      return $scope.jobCheckpointStats = data;\n    });\n  };\n  getOperatorCheckpointStats = function() {\n    return JobsService.getOperatorCheckpointStats($scope.nodeid).then(function(data) {\n      $scope.operatorCheckpointStats = data.operatorStats;\n      return $scope.subtasksCheckpointStats = data.subtasksStats;\n    });\n  };\n  getJobCheckpointStats();\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.operatorCheckpointStats)) {\n    getOperatorCheckpointStats();\n  }\n  return $scope.$on('reload', function(event) {\n    getJobCheckpointStats();\n    if ($scope.nodeid) {\n      return getOperatorCheckpointStats();\n    }\n  });\n}).controller('JobPlanBackPressureController', function($scope, JobsService) {\n  var getOperatorBackPressure;\n  getOperatorBackPressure = function() {\n    $scope.now = Date.now();\n    if ($scope.nodeid) {\n      return JobsService.getOperatorBackPressure($scope.nodeid).then(function(data) {\n        return $scope.backPressureOperatorStats[$scope.nodeid] = data;\n      });\n    }\n  };\n  getOperatorBackPressure();\n  return $scope.$on('reload', function(event) {\n    return getOperatorBackPressure();\n  });\n}).controller('JobTimelineVertexController', function($scope, $state, $stateParams, JobsService) {\n  var getVertex;\n  getVertex = function() {\n    return JobsService.getVertex($stateParams.vertexId).then(function(data) {\n      return $scope.vertex = data;\n    });\n  };\n  getVertex();\n  return $scope.$on('reload', function(event) {\n    return getVertex();\n  });\n}).controller('JobExceptionsController', function($scope, $state, $stateParams, JobsService) {\n  return JobsService.loadExceptions().then(function(data) {\n    return $scope.exceptions = data;\n  });\n}).controller('JobPropertiesController', function($scope, JobsService) {\n  return $scope.changeNode = function(nodeid) {\n    if (nodeid !== $scope.nodeid) {\n      $scope.nodeid = nodeid;\n      return JobsService.getNode(nodeid).then(function(data) {\n        return $scope.node = data;\n      });\n    } else {\n      $scope.nodeid = null;\n      return $scope.node = null;\n    }\n  };\n}).controller('JobPlanMetricsController', function($scope, JobsService, MetricsService) {\n  var loadMetrics;\n  $scope.dragging = false;\n  $scope.window = MetricsService.getWindow();\n  $scope.availableMetrics = null;\n  $scope.$on('$destroy', function() {\n    return MetricsService.unRegisterObserver();\n  });\n  loadMetrics = function() {\n    JobsService.getVertex($scope.nodeid).then(function(data) {\n      return $scope.vertex = data;\n    });\n    return MetricsService.getAvailableMetrics($scope.jobid, $scope.nodeid).then(function(data) {\n      $scope.availableMetrics = data;\n      $scope.metrics = MetricsService.getMetricsSetup($scope.jobid, $scope.nodeid).names;\n      return MetricsService.registerObserver($scope.jobid, $scope.nodeid, function(data) {\n        return $scope.$broadcast(\"metrics:data:update\", data.timestamp, data.values);\n      });\n    });\n  };\n  $scope.dropped = function(event, index, item, external, type) {\n    MetricsService.orderMetrics($scope.jobid, $scope.nodeid, item, index);\n    $scope.$broadcast(\"metrics:refresh\", item);\n    loadMetrics();\n    return false;\n  };\n  $scope.dragStart = function() {\n    return $scope.dragging = true;\n  };\n  $scope.dragEnd = function() {\n    return $scope.dragging = false;\n  };\n  $scope.addMetric = function(metric) {\n    MetricsService.addMetric($scope.jobid, $scope.nodeid, metric.id);\n    return loadMetrics();\n  };\n  $scope.removeMetric = function(metric) {\n    MetricsService.removeMetric($scope.jobid, $scope.nodeid, metric);\n    return loadMetrics();\n  };\n  $scope.setMetricSize = function(metric, size) {\n    MetricsService.setMetricSize($scope.jobid, $scope.nodeid, metric, size);\n    return loadMetrics();\n  };\n  $scope.getValues = function(metric) {\n    return MetricsService.getValues($scope.jobid, $scope.nodeid, metric);\n  };\n  $scope.$on('node:change', function(event, nodeid) {\n    if (!$scope.dragging) {\n      return loadMetrics();\n    }\n  });\n  if ($scope.nodeid) {\n    return loadMetrics();\n  }\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n# ----------------------------------------------\r\n\r\n.directive 'vertex', ($state) ->\r\n  template: \"<svg class='timeline secondary' width='0' height='0'></svg>\"\r\n\r\n  scope:\r\n    data: \"=\"\r\n\r\n  link: (scope, elem, attrs) ->\r\n    svgEl = elem.children()[0]\r\n\r\n    containerW = elem.width()\r\n    angular.element(svgEl).attr('width', containerW)\r\n\r\n    analyzeTime = (data) ->\r\n      d3.select(svgEl).selectAll(\"*\").remove()\r\n\r\n      testData = []\r\n\r\n      angular.forEach data.subtasks, (subtask, i) ->\r\n        times = [\r\n          {\r\n            label: \"Scheduled\"\r\n            color: \"#666\"\r\n            borderColor: \"#555\"\r\n            starting_time: subtask.timestamps[\"SCHEDULED\"]\r\n            ending_time: subtask.timestamps[\"DEPLOYING\"]\r\n            type: 'regular'\r\n          }\r\n          {\r\n            label: \"Deploying\"\r\n            color: \"#aaa\"\r\n            borderColor: \"#555\"\r\n            starting_time: subtask.timestamps[\"DEPLOYING\"]\r\n            ending_time: subtask.timestamps[\"RUNNING\"]\r\n            type: 'regular'\r\n          }\r\n        ]\r\n\r\n        if subtask.timestamps[\"FINISHED\"] > 0\r\n          times.push {\r\n            label: \"Running\"\r\n            color: \"#ddd\"\r\n            borderColor: \"#555\"\r\n            starting_time: subtask.timestamps[\"RUNNING\"]\r\n            ending_time: subtask.timestamps[\"FINISHED\"]\r\n            type: 'regular'\r\n          }\r\n\r\n        testData.push {\r\n          label: \"(#{subtask.subtask}) #{subtask.host}\"\r\n          times: times\r\n        }\r\n\r\n      chart = d3.timeline().stack()\r\n      .tickFormat({\r\n        format: d3.time.format(\"%L\")\r\n        # tickInterval: 1\r\n        tickSize: 1\r\n      })\r\n      .prefix(\"single\")\r\n      .labelFormat((label) ->\r\n        label\r\n      )\r\n      .margin({ left: 100, right: 0, top: 0, bottom: 0 })\r\n      .itemHeight(30)\r\n      .relativeTime()\r\n\r\n      svg = d3.select(svgEl)\r\n      .datum(testData)\r\n      .call(chart)\r\n\r\n    analyzeTime(scope.data)\r\n\r\n    return\r\n\r\n# ----------------------------------------------\r\n\r\n.directive 'timeline', ($state) ->\r\n  template: \"<svg class='timeline' width='0' height='0'></svg>\"\r\n\r\n  scope:\r\n    vertices: \"=\"\r\n    jobid: \"=\"\r\n\r\n  link: (scope, elem, attrs) ->\r\n    svgEl = elem.children()[0]\r\n\r\n    containerW = elem.width()\r\n    angular.element(svgEl).attr('width', containerW)\r\n\r\n    translateLabel = (label) ->\r\n      label.replace(\"&gt;\", \">\")\r\n\r\n    analyzeTime = (data) ->\r\n      d3.select(svgEl).selectAll(\"*\").remove()\r\n\r\n      testData = []\r\n\r\n      angular.forEach data, (vertex) ->\r\n        if vertex['start-time'] > -1\r\n          if vertex.type is 'scheduled'\r\n            testData.push\r\n              times: [\r\n                label: translateLabel(vertex.name)\r\n                color: \"#cccccc\"\r\n                borderColor: \"#555555\"\r\n                starting_time: vertex['start-time']\r\n                ending_time: vertex['end-time']\r\n                type: vertex.type\r\n              ]\r\n          else\r\n            testData.push\r\n              times: [\r\n                label: translateLabel(vertex.name)\r\n                color: \"#d9f1f7\"\r\n                borderColor: \"#62cdea\"\r\n                starting_time: vertex['start-time']\r\n                ending_time: vertex['end-time']\r\n                link: vertex.id\r\n                type: vertex.type\r\n              ]\r\n\r\n      chart = d3.timeline().stack().click((d, i, datum) ->\r\n        if d.link\r\n          $state.go \"single-job.timeline.vertex\", { jobid: scope.jobid, vertexId: d.link }\r\n\r\n      )\r\n      .tickFormat({\r\n        format: d3.time.format(\"%L\")\r\n        # tickTime: d3.time.second\r\n        # tickInterval: 0.5\r\n        tickSize: 1\r\n      })\r\n      .prefix(\"main\")\r\n      .margin({ left: 0, right: 0, top: 0, bottom: 0 })\r\n      .itemHeight(30)\r\n      .showBorderLine()\r\n      .showHourTimeline()\r\n\r\n      svg = d3.select(svgEl)\r\n      .datum(testData)\r\n      .call(chart)\r\n\r\n    scope.$watch attrs.vertices, (data) ->\r\n      analyzeTime(data) if data\r\n\r\n    return\r\n\r\n# ----------------------------------------------\r\n.directive 'split', () -> \r\n  return compile: (tElem, tAttrs) ->\r\n      Split(tElem.children(), (\r\n        sizes: [50, 50]\r\n        direction: 'vertical'\r\n      ))\r\n# ----------------------------------------------\r\n\r\n.directive 'jobPlan', ($timeout) ->\r\n  template: \"\r\n    <svg class='graph' width='500' height='400'><g /></svg>\r\n    <svg class='tmp' width='1' height='1'><g /></svg>\r\n    <div class='btn-group zoom-buttons'>\r\n      <a class='btn btn-default zoom-in' ng-click='zoomIn()'><i class='fa fa-plus' /></a>\r\n      <a class='btn btn-default zoom-out' ng-click='zoomOut()'><i class='fa fa-minus' /></a>\r\n    </div>\"\r\n\r\n  scope:\r\n    plan: '='\r\n    setNode: '&'\r\n\r\n  link: (scope, elem, attrs) ->\r\n    g = null\r\n    mainZoom = d3.behavior.zoom()\r\n    subgraphs = []\r\n    jobid = attrs.jobid\r\n\r\n    mainSvgElement = elem.children()[0]\r\n    mainG = elem.children().children()[0]\r\n    mainTmpElement = elem.children()[1]\r\n\r\n    d3mainSvg = d3.select(mainSvgElement)\r\n    d3mainSvgG = d3.select(mainG)\r\n    d3tmpSvg = d3.select(mainTmpElement)\r\n\r\n    # angular.element(mainG).empty()\r\n    # d3mainSvgG.selectAll(\"*\").remove()\r\n\r\n    containerW = elem.width()\r\n    angular.element(elem.children()[0]).width(containerW)\r\n\r\n    scope.zoomIn = ->\r\n      if mainZoom.scale() < 2.99\r\n\r\n        # Calculate and store new values in zoom object\r\n        translate = mainZoom.translate()\r\n        v1 = translate[0] * (mainZoom.scale() + 0.1 / (mainZoom.scale()))\r\n        v2 = translate[1] * (mainZoom.scale() + 0.1 / (mainZoom.scale()))\r\n        mainZoom.scale mainZoom.scale() + 0.1\r\n        mainZoom.translate [ v1, v2 ]\r\n\r\n        # Transform svg\r\n        d3mainSvgG.attr \"transform\", \"translate(\" + v1 + \",\" + v2 + \") scale(\" + mainZoom.scale() + \")\"\r\n\r\n    scope.zoomOut = ->\r\n      if mainZoom.scale() > 0.31\r\n\r\n        # Calculate and store new values in mainZoom object\r\n        mainZoom.scale mainZoom.scale() - 0.1\r\n        translate = mainZoom.translate()\r\n        v1 = translate[0] * (mainZoom.scale() - 0.1 / (mainZoom.scale()))\r\n        v2 = translate[1] * (mainZoom.scale() - 0.1 / (mainZoom.scale()))\r\n        mainZoom.translate [ v1, v2 ]\r\n\r\n        # Transform svg\r\n        d3mainSvgG.attr \"transform\", \"translate(\" + v1 + \",\" + v2 + \") scale(\" + mainZoom.scale() + \")\"\r\n\r\n    #create a label of an edge\r\n    createLabelEdge = (el) ->\r\n      labelValue = \"\"\r\n      if el.ship_strategy? or el.local_strategy?\r\n        labelValue += \"<div class='edge-label'>\"\r\n        labelValue += el.ship_strategy  if el.ship_strategy?\r\n        labelValue += \" (\" + el.temp_mode + \")\"  unless el.temp_mode is `undefined`\r\n        labelValue += \",<br>\" + el.local_strategy  unless el.local_strategy is `undefined`\r\n        labelValue += \"</div>\"\r\n      labelValue\r\n\r\n\r\n    # true, if the node is a special node from an iteration\r\n    isSpecialIterationNode = (info) ->\r\n      (info is \"partialSolution\" or info is \"nextPartialSolution\" or info is \"workset\" or info is \"nextWorkset\" or info is \"solutionSet\" or info is \"solutionDelta\")\r\n\r\n    getNodeType = (el, info) ->\r\n      if info is \"mirror\"\r\n        'node-mirror'\r\n\r\n      else if isSpecialIterationNode(info)\r\n        'node-iteration'\r\n\r\n      else\r\n        'node-normal'\r\n\r\n    # creates the label of a node, in info is stored, whether it is a special node (like a mirror in an iteration)\r\n    createLabelNode = (el, info, maxW, maxH) ->\r\n      # labelValue = \"<a href='#/jobs/\" + jobid + \"/vertex/\" + el.id + \"' class='node-label \" + getNodeType(el, info) + \"'>\"\r\n      labelValue = \"<div href='#/jobs/\" + jobid + \"/vertex/\" + el.id + \"' class='node-label \" + getNodeType(el, info) + \"'>\"\r\n\r\n      # Nodename\r\n      if info is \"mirror\"\r\n        labelValue += \"<h3 class='node-name'>Mirror of \" + el.operator + \"</h3>\"\r\n      else\r\n        labelValue += \"<h3 class='node-name'>\" + el.operator + \"</h3>\"\r\n      if el.description is \"\"\r\n        labelValue += \"\"\r\n      else\r\n        stepName = el.description\r\n\r\n        # clean stepName\r\n        stepName = shortenString(stepName)\r\n        labelValue += \"<h4 class='step-name'>\" + stepName + \"</h4>\"\r\n\r\n      # If this node is an \"iteration\" we need a different panel-body\r\n      if el.step_function?\r\n        labelValue += extendLabelNodeForIteration(el.id, maxW, maxH)\r\n      else\r\n\r\n        # Otherwise add infos\r\n        labelValue += \"<h5>\" + info + \" Node</h5>\"  if isSpecialIterationNode(info)\r\n        labelValue += \"<h5>Parallelism: \" + el.parallelism + \"</h5>\"  unless el.parallelism is \"\"\r\n        labelValue += \"<h5>Operation: \" + shortenString(el.operator_strategy) + \"</h5>\" unless el.operator is `undefined` or not el.operator_strategy\r\n      # labelValue += \"</a>\"\r\n      labelValue += \"</div>\"\r\n      labelValue\r\n\r\n    # Extends the label of a node with an additional svg Element to present the iteration.\r\n    extendLabelNodeForIteration = (id, maxW, maxH) ->\r\n      svgID = \"svg-\" + id\r\n\r\n      labelValue = \"<svg class='\" + svgID + \"' width=\" + maxW + \" height=\" + maxH + \"><g /></svg>\"\r\n      labelValue\r\n\r\n    # Split a string into multiple lines so that each line has less than 30 letters.\r\n    shortenString = (s) ->\r\n      # make sure that name does not contain a < (because of html)\r\n      if s.charAt(0) is \"<\"\r\n        s = s.replace(\"<\", \"&lt;\")\r\n        s = s.replace(\">\", \"&gt;\")\r\n      sbr = \"\"\r\n      while s.length > 30\r\n        sbr = sbr + s.substring(0, 30) + \"<br>\"\r\n        s = s.substring(30, s.length)\r\n      sbr = sbr + s\r\n      sbr\r\n\r\n    createNode = (g, data, el, isParent = false, maxW, maxH) ->\r\n      # create node, send additional informations about the node if it is a special one\r\n      if el.id is data.partial_solution\r\n        g.setNode el.id,\r\n          label: createLabelNode(el, \"partialSolution\", maxW, maxH)\r\n          labelType: 'html'\r\n          class: getNodeType(el, \"partialSolution\")\r\n\r\n      else if el.id is data.next_partial_solution\r\n        g.setNode el.id,\r\n          label: createLabelNode(el, \"nextPartialSolution\", maxW, maxH)\r\n          labelType: 'html'\r\n          class: getNodeType(el, \"nextPartialSolution\")\r\n\r\n      else if el.id is data.workset\r\n        g.setNode el.id,\r\n          label: createLabelNode(el, \"workset\", maxW, maxH)\r\n          labelType: 'html'\r\n          class: getNodeType(el, \"workset\")\r\n\r\n      else if el.id is data.next_workset\r\n        g.setNode el.id,\r\n          label: createLabelNode(el, \"nextWorkset\", maxW, maxH)\r\n          labelType: 'html'\r\n          class: getNodeType(el, \"nextWorkset\")\r\n\r\n      else if el.id is data.solution_set\r\n        g.setNode el.id,\r\n          label: createLabelNode(el, \"solutionSet\", maxW, maxH)\r\n          labelType: 'html'\r\n          class: getNodeType(el, \"solutionSet\")\r\n\r\n      else if el.id is data.solution_delta\r\n        g.setNode el.id,\r\n          label: createLabelNode(el, \"solutionDelta\", maxW, maxH)\r\n          labelType: 'html'\r\n          class: getNodeType(el, \"solutionDelta\")\r\n\r\n      else\r\n        g.setNode el.id,\r\n          label: createLabelNode(el, \"\", maxW, maxH)\r\n          labelType: 'html'\r\n          class: getNodeType(el, \"\")\r\n\r\n    createEdge = (g, data, el, existingNodes, pred) ->\r\n      g.setEdge pred.id, el.id,\r\n        label: createLabelEdge(pred)\r\n        labelType: 'html'\r\n        arrowhead: 'normal'\r\n\r\n    loadJsonToDagre = (g, data) ->\r\n      existingNodes = []\r\n\r\n      if data.nodes?\r\n        # This is the normal json data\r\n        toIterate = data.nodes\r\n\r\n      else\r\n        # This is an iteration, we now store special iteration nodes if possible\r\n        toIterate = data.step_function\r\n        isParent = true\r\n\r\n      for el in toIterate\r\n        maxW = 0\r\n        maxH = 0\r\n\r\n        if el.step_function\r\n          sg = new dagreD3.graphlib.Graph({ multigraph: true, compound: true }).setGraph({\r\n            nodesep: 20\r\n            edgesep: 0\r\n            ranksep: 20\r\n            rankdir: \"LR\"\r\n            marginx: 10\r\n            marginy: 10\r\n            })\r\n\r\n          subgraphs[el.id] = sg\r\n\r\n          loadJsonToDagre(sg, el)\r\n\r\n          r = new dagreD3.render()\r\n          d3tmpSvg.select('g').call(r, sg)\r\n          maxW = sg.graph().width\r\n          maxH = sg.graph().height\r\n\r\n          angular.element(mainTmpElement).empty()\r\n\r\n        createNode(g, data, el, isParent, maxW, maxH)\r\n\r\n        existingNodes.push el.id\r\n\r\n        # create edges from inputs to current node\r\n        if el.inputs?\r\n          for pred in el.inputs\r\n            createEdge(g, data, el, existingNodes, pred)\r\n\r\n      g\r\n\r\n    # searches in the global JSONData for the node with the given id\r\n    searchForNode = (data, nodeID) ->\r\n      for i of data.nodes\r\n        el = data.nodes[i]\r\n        return el  if el.id is nodeID\r\n\r\n        # look for nodes that are in iterations\r\n        if el.step_function?\r\n          for j of el.step_function\r\n            return el.step_function[j]  if el.step_function[j].id is nodeID\r\n\r\n    drawGraph = (data) ->\r\n      g = new dagreD3.graphlib.Graph({ multigraph: true, compound: true }).setGraph({\r\n        nodesep: 70\r\n        edgesep: 0\r\n        ranksep: 50\r\n        rankdir: \"LR\"\r\n        marginx: 40\r\n        marginy: 40\r\n        })\r\n\r\n      loadJsonToDagre(g, data)\r\n\r\n      renderer = new dagreD3.render()\r\n      d3mainSvgG.call(renderer, g)\r\n\r\n      for i, sg of subgraphs\r\n        d3mainSvg.select('svg.svg-' + i + ' g').call(renderer, sg)\r\n\r\n      newScale = 0.5\r\n\r\n      xCenterOffset = Math.floor((angular.element(mainSvgElement).width() - g.graph().width * newScale) / 2)\r\n      yCenterOffset = Math.floor((angular.element(mainSvgElement).height() - g.graph().height * newScale) / 2)\r\n\r\n      mainZoom.scale(newScale).translate([xCenterOffset, yCenterOffset])\r\n\r\n      d3mainSvgG.attr(\"transform\", \"translate(\" + xCenterOffset + \", \" + yCenterOffset + \") scale(\" + mainZoom.scale() + \")\")\r\n\r\n      mainZoom.on(\"zoom\", ->\r\n        ev = d3.event\r\n        d3mainSvgG.attr \"transform\", \"translate(\" + ev.translate + \") scale(\" + ev.scale + \")\"\r\n      )\r\n      mainZoom(d3mainSvg)\r\n\r\n      d3mainSvgG.selectAll('.node').on 'click', (d) ->\r\n        scope.setNode({ nodeid: d })\r\n\r\n    scope.$watch attrs.plan, (newPlan) ->\r\n      drawGraph(newPlan) if newPlan\r\n\r\n    return\r\n","angular.module('flinkApp').directive('vertex', function($state) {\n  return {\n    template: \"<svg class='timeline secondary' width='0' height='0'></svg>\",\n    scope: {\n      data: \"=\"\n    },\n    link: function(scope, elem, attrs) {\n      var analyzeTime, containerW, svgEl;\n      svgEl = elem.children()[0];\n      containerW = elem.width();\n      angular.element(svgEl).attr('width', containerW);\n      analyzeTime = function(data) {\n        var chart, svg, testData;\n        d3.select(svgEl).selectAll(\"*\").remove();\n        testData = [];\n        angular.forEach(data.subtasks, function(subtask, i) {\n          var times;\n          times = [\n            {\n              label: \"Scheduled\",\n              color: \"#666\",\n              borderColor: \"#555\",\n              starting_time: subtask.timestamps[\"SCHEDULED\"],\n              ending_time: subtask.timestamps[\"DEPLOYING\"],\n              type: 'regular'\n            }, {\n              label: \"Deploying\",\n              color: \"#aaa\",\n              borderColor: \"#555\",\n              starting_time: subtask.timestamps[\"DEPLOYING\"],\n              ending_time: subtask.timestamps[\"RUNNING\"],\n              type: 'regular'\n            }\n          ];\n          if (subtask.timestamps[\"FINISHED\"] > 0) {\n            times.push({\n              label: \"Running\",\n              color: \"#ddd\",\n              borderColor: \"#555\",\n              starting_time: subtask.timestamps[\"RUNNING\"],\n              ending_time: subtask.timestamps[\"FINISHED\"],\n              type: 'regular'\n            });\n          }\n          return testData.push({\n            label: \"(\" + subtask.subtask + \") \" + subtask.host,\n            times: times\n          });\n        });\n        chart = d3.timeline().stack().tickFormat({\n          format: d3.time.format(\"%L\"),\n          tickSize: 1\n        }).prefix(\"single\").labelFormat(function(label) {\n          return label;\n        }).margin({\n          left: 100,\n          right: 0,\n          top: 0,\n          bottom: 0\n        }).itemHeight(30).relativeTime();\n        return svg = d3.select(svgEl).datum(testData).call(chart);\n      };\n      analyzeTime(scope.data);\n    }\n  };\n}).directive('timeline', function($state) {\n  return {\n    template: \"<svg class='timeline' width='0' height='0'></svg>\",\n    scope: {\n      vertices: \"=\",\n      jobid: \"=\"\n    },\n    link: function(scope, elem, attrs) {\n      var analyzeTime, containerW, svgEl, translateLabel;\n      svgEl = elem.children()[0];\n      containerW = elem.width();\n      angular.element(svgEl).attr('width', containerW);\n      translateLabel = function(label) {\n        return label.replace(\"&gt;\", \">\");\n      };\n      analyzeTime = function(data) {\n        var chart, svg, testData;\n        d3.select(svgEl).selectAll(\"*\").remove();\n        testData = [];\n        angular.forEach(data, function(vertex) {\n          if (vertex['start-time'] > -1) {\n            if (vertex.type === 'scheduled') {\n              return testData.push({\n                times: [\n                  {\n                    label: translateLabel(vertex.name),\n                    color: \"#cccccc\",\n                    borderColor: \"#555555\",\n                    starting_time: vertex['start-time'],\n                    ending_time: vertex['end-time'],\n                    type: vertex.type\n                  }\n                ]\n              });\n            } else {\n              return testData.push({\n                times: [\n                  {\n                    label: translateLabel(vertex.name),\n                    color: \"#d9f1f7\",\n                    borderColor: \"#62cdea\",\n                    starting_time: vertex['start-time'],\n                    ending_time: vertex['end-time'],\n                    link: vertex.id,\n                    type: vertex.type\n                  }\n                ]\n              });\n            }\n          }\n        });\n        chart = d3.timeline().stack().click(function(d, i, datum) {\n          if (d.link) {\n            return $state.go(\"single-job.timeline.vertex\", {\n              jobid: scope.jobid,\n              vertexId: d.link\n            });\n          }\n        }).tickFormat({\n          format: d3.time.format(\"%L\"),\n          tickSize: 1\n        }).prefix(\"main\").margin({\n          left: 0,\n          right: 0,\n          top: 0,\n          bottom: 0\n        }).itemHeight(30).showBorderLine().showHourTimeline();\n        return svg = d3.select(svgEl).datum(testData).call(chart);\n      };\n      scope.$watch(attrs.vertices, function(data) {\n        if (data) {\n          return analyzeTime(data);\n        }\n      });\n    }\n  };\n}).directive('split', function() {\n  return {\n    compile: function(tElem, tAttrs) {\n      return Split(tElem.children(), {\n        sizes: [50, 50],\n        direction: 'vertical'\n      });\n    }\n  };\n}).directive('jobPlan', function($timeout) {\n  return {\n    template: \"<svg class='graph' width='500' height='400'><g /></svg> <svg class='tmp' width='1' height='1'><g /></svg> <div class='btn-group zoom-buttons'> <a class='btn btn-default zoom-in' ng-click='zoomIn()'><i class='fa fa-plus' /></a> <a class='btn btn-default zoom-out' ng-click='zoomOut()'><i class='fa fa-minus' /></a> </div>\",\n    scope: {\n      plan: '=',\n      setNode: '&'\n    },\n    link: function(scope, elem, attrs) {\n      var containerW, createEdge, createLabelEdge, createLabelNode, createNode, d3mainSvg, d3mainSvgG, d3tmpSvg, drawGraph, extendLabelNodeForIteration, g, getNodeType, isSpecialIterationNode, jobid, loadJsonToDagre, mainG, mainSvgElement, mainTmpElement, mainZoom, searchForNode, shortenString, subgraphs;\n      g = null;\n      mainZoom = d3.behavior.zoom();\n      subgraphs = [];\n      jobid = attrs.jobid;\n      mainSvgElement = elem.children()[0];\n      mainG = elem.children().children()[0];\n      mainTmpElement = elem.children()[1];\n      d3mainSvg = d3.select(mainSvgElement);\n      d3mainSvgG = d3.select(mainG);\n      d3tmpSvg = d3.select(mainTmpElement);\n      containerW = elem.width();\n      angular.element(elem.children()[0]).width(containerW);\n      scope.zoomIn = function() {\n        var translate, v1, v2;\n        if (mainZoom.scale() < 2.99) {\n          translate = mainZoom.translate();\n          v1 = translate[0] * (mainZoom.scale() + 0.1 / (mainZoom.scale()));\n          v2 = translate[1] * (mainZoom.scale() + 0.1 / (mainZoom.scale()));\n          mainZoom.scale(mainZoom.scale() + 0.1);\n          mainZoom.translate([v1, v2]);\n          return d3mainSvgG.attr(\"transform\", \"translate(\" + v1 + \",\" + v2 + \") scale(\" + mainZoom.scale() + \")\");\n        }\n      };\n      scope.zoomOut = function() {\n        var translate, v1, v2;\n        if (mainZoom.scale() > 0.31) {\n          mainZoom.scale(mainZoom.scale() - 0.1);\n          translate = mainZoom.translate();\n          v1 = translate[0] * (mainZoom.scale() - 0.1 / (mainZoom.scale()));\n          v2 = translate[1] * (mainZoom.scale() - 0.1 / (mainZoom.scale()));\n          mainZoom.translate([v1, v2]);\n          return d3mainSvgG.attr(\"transform\", \"translate(\" + v1 + \",\" + v2 + \") scale(\" + mainZoom.scale() + \")\");\n        }\n      };\n      createLabelEdge = function(el) {\n        var labelValue;\n        labelValue = \"\";\n        if ((el.ship_strategy != null) || (el.local_strategy != null)) {\n          labelValue += \"<div class='edge-label'>\";\n          if (el.ship_strategy != null) {\n            labelValue += el.ship_strategy;\n          }\n          if (el.temp_mode !== undefined) {\n            labelValue += \" (\" + el.temp_mode + \")\";\n          }\n          if (el.local_strategy !== undefined) {\n            labelValue += \",<br>\" + el.local_strategy;\n          }\n          labelValue += \"</div>\";\n        }\n        return labelValue;\n      };\n      isSpecialIterationNode = function(info) {\n        return info === \"partialSolution\" || info === \"nextPartialSolution\" || info === \"workset\" || info === \"nextWorkset\" || info === \"solutionSet\" || info === \"solutionDelta\";\n      };\n      getNodeType = function(el, info) {\n        if (info === \"mirror\") {\n          return 'node-mirror';\n        } else if (isSpecialIterationNode(info)) {\n          return 'node-iteration';\n        } else {\n          return 'node-normal';\n        }\n      };\n      createLabelNode = function(el, info, maxW, maxH) {\n        var labelValue, stepName;\n        labelValue = \"<div href='#/jobs/\" + jobid + \"/vertex/\" + el.id + \"' class='node-label \" + getNodeType(el, info) + \"'>\";\n        if (info === \"mirror\") {\n          labelValue += \"<h3 class='node-name'>Mirror of \" + el.operator + \"</h3>\";\n        } else {\n          labelValue += \"<h3 class='node-name'>\" + el.operator + \"</h3>\";\n        }\n        if (el.description === \"\") {\n          labelValue += \"\";\n        } else {\n          stepName = el.description;\n          stepName = shortenString(stepName);\n          labelValue += \"<h4 class='step-name'>\" + stepName + \"</h4>\";\n        }\n        if (el.step_function != null) {\n          labelValue += extendLabelNodeForIteration(el.id, maxW, maxH);\n        } else {\n          if (isSpecialIterationNode(info)) {\n            labelValue += \"<h5>\" + info + \" Node</h5>\";\n          }\n          if (el.parallelism !== \"\") {\n            labelValue += \"<h5>Parallelism: \" + el.parallelism + \"</h5>\";\n          }\n          if (!(el.operator === undefined || !el.operator_strategy)) {\n            labelValue += \"<h5>Operation: \" + shortenString(el.operator_strategy) + \"</h5>\";\n          }\n        }\n        labelValue += \"</div>\";\n        return labelValue;\n      };\n      extendLabelNodeForIteration = function(id, maxW, maxH) {\n        var labelValue, svgID;\n        svgID = \"svg-\" + id;\n        labelValue = \"<svg class='\" + svgID + \"' width=\" + maxW + \" height=\" + maxH + \"><g /></svg>\";\n        return labelValue;\n      };\n      shortenString = function(s) {\n        var sbr;\n        if (s.charAt(0) === \"<\") {\n          s = s.replace(\"<\", \"&lt;\");\n          s = s.replace(\">\", \"&gt;\");\n        }\n        sbr = \"\";\n        while (s.length > 30) {\n          sbr = sbr + s.substring(0, 30) + \"<br>\";\n          s = s.substring(30, s.length);\n        }\n        sbr = sbr + s;\n        return sbr;\n      };\n      createNode = function(g, data, el, isParent, maxW, maxH) {\n        if (isParent == null) {\n          isParent = false;\n        }\n        if (el.id === data.partial_solution) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"partialSolution\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"partialSolution\")\n          });\n        } else if (el.id === data.next_partial_solution) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"nextPartialSolution\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"nextPartialSolution\")\n          });\n        } else if (el.id === data.workset) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"workset\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"workset\")\n          });\n        } else if (el.id === data.next_workset) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"nextWorkset\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"nextWorkset\")\n          });\n        } else if (el.id === data.solution_set) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"solutionSet\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"solutionSet\")\n          });\n        } else if (el.id === data.solution_delta) {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"solutionDelta\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"solutionDelta\")\n          });\n        } else {\n          return g.setNode(el.id, {\n            label: createLabelNode(el, \"\", maxW, maxH),\n            labelType: 'html',\n            \"class\": getNodeType(el, \"\")\n          });\n        }\n      };\n      createEdge = function(g, data, el, existingNodes, pred) {\n        return g.setEdge(pred.id, el.id, {\n          label: createLabelEdge(pred),\n          labelType: 'html',\n          arrowhead: 'normal'\n        });\n      };\n      loadJsonToDagre = function(g, data) {\n        var el, existingNodes, isParent, k, l, len, len1, maxH, maxW, pred, r, ref, sg, toIterate;\n        existingNodes = [];\n        if (data.nodes != null) {\n          toIterate = data.nodes;\n        } else {\n          toIterate = data.step_function;\n          isParent = true;\n        }\n        for (k = 0, len = toIterate.length; k < len; k++) {\n          el = toIterate[k];\n          maxW = 0;\n          maxH = 0;\n          if (el.step_function) {\n            sg = new dagreD3.graphlib.Graph({\n              multigraph: true,\n              compound: true\n            }).setGraph({\n              nodesep: 20,\n              edgesep: 0,\n              ranksep: 20,\n              rankdir: \"LR\",\n              marginx: 10,\n              marginy: 10\n            });\n            subgraphs[el.id] = sg;\n            loadJsonToDagre(sg, el);\n            r = new dagreD3.render();\n            d3tmpSvg.select('g').call(r, sg);\n            maxW = sg.graph().width;\n            maxH = sg.graph().height;\n            angular.element(mainTmpElement).empty();\n          }\n          createNode(g, data, el, isParent, maxW, maxH);\n          existingNodes.push(el.id);\n          if (el.inputs != null) {\n            ref = el.inputs;\n            for (l = 0, len1 = ref.length; l < len1; l++) {\n              pred = ref[l];\n              createEdge(g, data, el, existingNodes, pred);\n            }\n          }\n        }\n        return g;\n      };\n      searchForNode = function(data, nodeID) {\n        var el, i, j;\n        for (i in data.nodes) {\n          el = data.nodes[i];\n          if (el.id === nodeID) {\n            return el;\n          }\n          if (el.step_function != null) {\n            for (j in el.step_function) {\n              if (el.step_function[j].id === nodeID) {\n                return el.step_function[j];\n              }\n            }\n          }\n        }\n      };\n      drawGraph = function(data) {\n        var i, newScale, renderer, sg, xCenterOffset, yCenterOffset;\n        g = new dagreD3.graphlib.Graph({\n          multigraph: true,\n          compound: true\n        }).setGraph({\n          nodesep: 70,\n          edgesep: 0,\n          ranksep: 50,\n          rankdir: \"LR\",\n          marginx: 40,\n          marginy: 40\n        });\n        loadJsonToDagre(g, data);\n        renderer = new dagreD3.render();\n        d3mainSvgG.call(renderer, g);\n        for (i in subgraphs) {\n          sg = subgraphs[i];\n          d3mainSvg.select('svg.svg-' + i + ' g').call(renderer, sg);\n        }\n        newScale = 0.5;\n        xCenterOffset = Math.floor((angular.element(mainSvgElement).width() - g.graph().width * newScale) / 2);\n        yCenterOffset = Math.floor((angular.element(mainSvgElement).height() - g.graph().height * newScale) / 2);\n        mainZoom.scale(newScale).translate([xCenterOffset, yCenterOffset]);\n        d3mainSvgG.attr(\"transform\", \"translate(\" + xCenterOffset + \", \" + yCenterOffset + \") scale(\" + mainZoom.scale() + \")\");\n        mainZoom.on(\"zoom\", function() {\n          var ev;\n          ev = d3.event;\n          return d3mainSvgG.attr(\"transform\", \"translate(\" + ev.translate + \") scale(\" + ev.scale + \")\");\n        });\n        mainZoom(d3mainSvg);\n        return d3mainSvgG.selectAll('.node').on('click', function(d) {\n          return scope.setNode({\n            nodeid: d\n          });\n        });\n      };\n      scope.$watch(attrs.plan, function(newPlan) {\n        if (newPlan) {\n          return drawGraph(newPlan);\n        }\n      });\n    }\n  };\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.service 'JobsService', ($http, flinkConfig, $log, amMoment, $q, $timeout) ->\r\n  currentJob = null\r\n  currentPlan = null\r\n\r\n  deferreds = {}\r\n  jobs = {\r\n    running: []\r\n    finished: []\r\n    cancelled: []\r\n    failed: []\r\n  }\r\n\r\n  jobObservers = []\r\n\r\n  notifyObservers = ->\r\n    angular.forEach jobObservers, (callback) ->\r\n      callback()\r\n\r\n  @registerObserver = (callback) ->\r\n    jobObservers.push(callback)\r\n\r\n  @unRegisterObserver = (callback) ->\r\n    index = jobObservers.indexOf(callback)\r\n    jobObservers.splice(index, 1)\r\n\r\n  @stateList = ->\r\n    [ \r\n      # 'CREATED'\r\n      'SCHEDULED'\r\n      'DEPLOYING'\r\n      'RUNNING'\r\n      'FINISHED'\r\n      'FAILED'\r\n      'CANCELING'\r\n      'CANCELED'\r\n    ]\r\n\r\n  @translateLabelState = (state) ->\r\n    switch state.toLowerCase()\r\n      when 'finished' then 'success'\r\n      when 'failed' then 'danger'\r\n      when 'scheduled' then 'default'\r\n      when 'deploying' then 'info'\r\n      when 'running' then 'primary'\r\n      when 'canceling' then 'warning'\r\n      when 'pending' then 'info'\r\n      when 'total' then 'black'\r\n      else 'default'\r\n\r\n  @setEndTimes = (list) ->\r\n    angular.forEach list, (item, jobKey) ->\r\n      unless item['end-time'] > -1\r\n        item['end-time'] = item['start-time'] + item['duration']\r\n\r\n  @processVertices = (data) ->\r\n    angular.forEach data.vertices, (vertex, i) ->\r\n      vertex.type = 'regular'\r\n\r\n    data.vertices.unshift({\r\n      name: 'Scheduled'\r\n      'start-time': data.timestamps['CREATED']\r\n      'end-time': data.timestamps['CREATED'] + 1\r\n      type: 'scheduled'\r\n    })\r\n\r\n  @listJobs = ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get flinkConfig.jobServer + \"joboverview\"\r\n    .success (data, status, headers, config) =>\r\n      angular.forEach data, (list, listKey) =>\r\n        switch listKey\r\n          when 'running' then jobs.running = @setEndTimes(list)\r\n          when 'finished' then jobs.finished = @setEndTimes(list)\r\n          when 'cancelled' then jobs.cancelled = @setEndTimes(list)\r\n          when 'failed' then jobs.failed = @setEndTimes(list)\r\n\r\n      deferred.resolve(jobs)\r\n      notifyObservers()\r\n\r\n    deferred.promise\r\n\r\n  @getJobs = (type) ->\r\n    jobs[type]\r\n\r\n  @getAllJobs = ->\r\n    jobs\r\n\r\n  @loadJob = (jobid) ->\r\n    currentJob = null\r\n    deferreds.job = $q.defer()\r\n\r\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid\r\n    .success (data, status, headers, config) =>\r\n      @setEndTimes(data.vertices)\r\n      @processVertices(data)\r\n\r\n      $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/config\"\r\n      .success (jobConfig) ->\r\n        data = angular.extend(data, jobConfig)\r\n\r\n        currentJob = data\r\n\r\n        deferreds.job.resolve(currentJob)\r\n\r\n    deferreds.job.promise\r\n\r\n  @getNode = (nodeid) ->\r\n    seekNode = (nodeid, data) ->\r\n      for node in data\r\n        return node if node.id is nodeid\r\n        sub = seekNode(nodeid, node.step_function) if node.step_function\r\n        return sub if sub\r\n\r\n      null\r\n\r\n    deferred = $q.defer()\r\n\r\n    deferreds.job.promise.then (data) =>\r\n      foundNode = seekNode(nodeid, currentJob.plan.nodes)\r\n\r\n      foundNode.vertex = @seekVertex(nodeid)\r\n\r\n      deferred.resolve(foundNode)\r\n\r\n    deferred.promise\r\n\r\n  @seekVertex = (nodeid) ->\r\n    for vertex in currentJob.vertices\r\n      return vertex if vertex.id is nodeid\r\n\r\n    return null\r\n\r\n  @getVertex = (vertexid) ->\r\n    deferred = $q.defer()\r\n\r\n    deferreds.job.promise.then (data) =>\r\n      vertex = @seekVertex(vertexid)\r\n\r\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/subtasktimes\"\r\n      .success (data) =>\r\n        # TODO: change to subtasktimes\r\n        vertex.subtasks = data.subtasks\r\n\r\n        deferred.resolve(vertex)\r\n\r\n    deferred.promise\r\n\r\n  @getSubtasks = (vertexid) ->\r\n    deferred = $q.defer()\r\n\r\n    deferreds.job.promise.then (data) =>\r\n      # vertex = @seekVertex(vertexid)\r\n\r\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid\r\n      .success (data) ->\r\n        subtasks = data.subtasks\r\n\r\n        deferred.resolve(subtasks)\r\n\r\n    deferred.promise\r\n\r\n  @getTaskManagers = (vertexid) ->\r\n    deferred = $q.defer()\r\n\r\n    deferreds.job.promise.then (data) =>\r\n      # vertex = @seekVertex(vertexid)\r\n\r\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/taskmanagers\"\r\n      .success (data) ->\r\n        taskmanagers = data.taskmanagers\r\n\r\n        deferred.resolve(taskmanagers)\r\n\r\n    deferred.promise\r\n\r\n  @getAccumulators = (vertexid) ->\r\n    deferred = $q.defer()\r\n\r\n    deferreds.job.promise.then (data) =>\r\n      # vertex = @seekVertex(vertexid)\r\n\r\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/accumulators\"\r\n      .success (data) ->\r\n        accumulators = data['user-accumulators']\r\n\r\n        $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/subtasks/accumulators\"\r\n        .success (data) ->\r\n          subtaskAccumulators = data.subtasks\r\n\r\n          deferred.resolve({ main: accumulators, subtasks: subtaskAccumulators })\r\n\r\n    deferred.promise\r\n\r\n  # Job-level checkpoint stats\r\n  @getJobCheckpointStats = (jobid) ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/checkpoints\"\r\n    .success (data, status, headers, config) =>\r\n      if (angular.equals({}, data))\r\n        deferred.resolve(deferred.resolve(null))\r\n      else\r\n        deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  # Operator-level checkpoint stats\r\n  @getOperatorCheckpointStats = (vertexid) ->\r\n    deferred = $q.defer()\r\n\r\n    deferreds.job.promise.then (data) =>\r\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/checkpoints\"\r\n      .success (data) ->\r\n        # If no data available, we are done.\r\n        if (angular.equals({}, data))\r\n          deferred.resolve({ operatorStats: null, subtasksStats: null })\r\n        else\r\n          operatorStats = { id: data['id'], timestamp: data['timestamp'], duration: data['duration'], size: data['size'] }\r\n\r\n          if (angular.equals({}, data['subtasks']))\r\n            deferred.resolve({ operatorStats: operatorStats, subtasksStats: null })\r\n          else\r\n            subtaskStats = data['subtasks']\r\n            deferred.resolve({ operatorStats: operatorStats, subtasksStats: subtaskStats })\r\n\r\n    deferred.promise\r\n\r\n  # Operator-level back pressure stats\r\n  @getOperatorBackPressure = (vertexid) ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/backpressure\"\r\n    .success (data) =>\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @translateBackPressureLabelState = (state) ->\r\n    switch state.toLowerCase()\r\n      when 'in-progress' then 'danger'\r\n      when 'ok' then 'success'\r\n      when 'low' then 'warning'\r\n      when 'high' then 'danger'\r\n      else 'default'\r\n\r\n  @loadExceptions = ->\r\n    deferred = $q.defer()\r\n\r\n    deferreds.job.promise.then (data) =>\r\n\r\n      $http.get flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/exceptions\"\r\n      .success (exceptions) ->\r\n        currentJob.exceptions = exceptions\r\n\r\n        deferred.resolve(exceptions)\r\n\r\n    deferred.promise\r\n\r\n  @cancelJob = (jobid) ->\r\n    # uses the non REST-compliant GET yarn-cancel handler which is available in addition to the\r\n    # proper \"DELETE jobs/<jobid>/\"\r\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/yarn-cancel\"\r\n\r\n  @stopJob = (jobid) ->\r\n    # uses the non REST-compliant GET yarn-cancel handler which is available in addition to the\r\n    # proper \"DELETE jobs/<jobid>/\"\r\n    $http.get \"jobs/\" + jobid + \"/yarn-stop\"\r\n\r\n  @\r\n","angular.module('flinkApp').service('JobsService', function($http, flinkConfig, $log, amMoment, $q, $timeout) {\n  var currentJob, currentPlan, deferreds, jobObservers, jobs, notifyObservers;\n  currentJob = null;\n  currentPlan = null;\n  deferreds = {};\n  jobs = {\n    running: [],\n    finished: [],\n    cancelled: [],\n    failed: []\n  };\n  jobObservers = [];\n  notifyObservers = function() {\n    return angular.forEach(jobObservers, function(callback) {\n      return callback();\n    });\n  };\n  this.registerObserver = function(callback) {\n    return jobObservers.push(callback);\n  };\n  this.unRegisterObserver = function(callback) {\n    var index;\n    index = jobObservers.indexOf(callback);\n    return jobObservers.splice(index, 1);\n  };\n  this.stateList = function() {\n    return ['SCHEDULED', 'DEPLOYING', 'RUNNING', 'FINISHED', 'FAILED', 'CANCELING', 'CANCELED'];\n  };\n  this.translateLabelState = function(state) {\n    switch (state.toLowerCase()) {\n      case 'finished':\n        return 'success';\n      case 'failed':\n        return 'danger';\n      case 'scheduled':\n        return 'default';\n      case 'deploying':\n        return 'info';\n      case 'running':\n        return 'primary';\n      case 'canceling':\n        return 'warning';\n      case 'pending':\n        return 'info';\n      case 'total':\n        return 'black';\n      default:\n        return 'default';\n    }\n  };\n  this.setEndTimes = function(list) {\n    return angular.forEach(list, function(item, jobKey) {\n      if (!(item['end-time'] > -1)) {\n        return item['end-time'] = item['start-time'] + item['duration'];\n      }\n    });\n  };\n  this.processVertices = function(data) {\n    angular.forEach(data.vertices, function(vertex, i) {\n      return vertex.type = 'regular';\n    });\n    return data.vertices.unshift({\n      name: 'Scheduled',\n      'start-time': data.timestamps['CREATED'],\n      'end-time': data.timestamps['CREATED'] + 1,\n      type: 'scheduled'\n    });\n  };\n  this.listJobs = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"joboverview\").success((function(_this) {\n      return function(data, status, headers, config) {\n        angular.forEach(data, function(list, listKey) {\n          switch (listKey) {\n            case 'running':\n              return jobs.running = _this.setEndTimes(list);\n            case 'finished':\n              return jobs.finished = _this.setEndTimes(list);\n            case 'cancelled':\n              return jobs.cancelled = _this.setEndTimes(list);\n            case 'failed':\n              return jobs.failed = _this.setEndTimes(list);\n          }\n        });\n        deferred.resolve(jobs);\n        return notifyObservers();\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getJobs = function(type) {\n    return jobs[type];\n  };\n  this.getAllJobs = function() {\n    return jobs;\n  };\n  this.loadJob = function(jobid) {\n    currentJob = null;\n    deferreds.job = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobs/\" + jobid).success((function(_this) {\n      return function(data, status, headers, config) {\n        _this.setEndTimes(data.vertices);\n        _this.processVertices(data);\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/config\").success(function(jobConfig) {\n          data = angular.extend(data, jobConfig);\n          currentJob = data;\n          return deferreds.job.resolve(currentJob);\n        });\n      };\n    })(this));\n    return deferreds.job.promise;\n  };\n  this.getNode = function(nodeid) {\n    var deferred, seekNode;\n    seekNode = function(nodeid, data) {\n      var j, len, node, sub;\n      for (j = 0, len = data.length; j < len; j++) {\n        node = data[j];\n        if (node.id === nodeid) {\n          return node;\n        }\n        if (node.step_function) {\n          sub = seekNode(nodeid, node.step_function);\n        }\n        if (sub) {\n          return sub;\n        }\n      }\n      return null;\n    };\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        var foundNode;\n        foundNode = seekNode(nodeid, currentJob.plan.nodes);\n        foundNode.vertex = _this.seekVertex(nodeid);\n        return deferred.resolve(foundNode);\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.seekVertex = function(nodeid) {\n    var j, len, ref, vertex;\n    ref = currentJob.vertices;\n    for (j = 0, len = ref.length; j < len; j++) {\n      vertex = ref[j];\n      if (vertex.id === nodeid) {\n        return vertex;\n      }\n    }\n    return null;\n  };\n  this.getVertex = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        var vertex;\n        vertex = _this.seekVertex(vertexid);\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/subtasktimes\").success(function(data) {\n          vertex.subtasks = data.subtasks;\n          return deferred.resolve(vertex);\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getSubtasks = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid).success(function(data) {\n          var subtasks;\n          subtasks = data.subtasks;\n          return deferred.resolve(subtasks);\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getTaskManagers = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/taskmanagers\").success(function(data) {\n          var taskmanagers;\n          taskmanagers = data.taskmanagers;\n          return deferred.resolve(taskmanagers);\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getAccumulators = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/accumulators\").success(function(data) {\n          var accumulators;\n          accumulators = data['user-accumulators'];\n          return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/subtasks/accumulators\").success(function(data) {\n            var subtaskAccumulators;\n            subtaskAccumulators = data.subtasks;\n            return deferred.resolve({\n              main: accumulators,\n              subtasks: subtaskAccumulators\n            });\n          });\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getJobCheckpointStats = function(jobid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/checkpoints\").success((function(_this) {\n      return function(data, status, headers, config) {\n        if (angular.equals({}, data)) {\n          return deferred.resolve(deferred.resolve(null));\n        } else {\n          return deferred.resolve(data);\n        }\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getOperatorCheckpointStats = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/checkpoints\").success(function(data) {\n          var operatorStats, subtaskStats;\n          if (angular.equals({}, data)) {\n            return deferred.resolve({\n              operatorStats: null,\n              subtasksStats: null\n            });\n          } else {\n            operatorStats = {\n              id: data['id'],\n              timestamp: data['timestamp'],\n              duration: data['duration'],\n              size: data['size']\n            };\n            if (angular.equals({}, data['subtasks'])) {\n              return deferred.resolve({\n                operatorStats: operatorStats,\n                subtasksStats: null\n              });\n            } else {\n              subtaskStats = data['subtasks'];\n              return deferred.resolve({\n                operatorStats: operatorStats,\n                subtasksStats: subtaskStats\n              });\n            }\n          }\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.getOperatorBackPressure = function(vertexid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/backpressure\").success((function(_this) {\n      return function(data) {\n        return deferred.resolve(data);\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.translateBackPressureLabelState = function(state) {\n    switch (state.toLowerCase()) {\n      case 'in-progress':\n        return 'danger';\n      case 'ok':\n        return 'success';\n      case 'low':\n        return 'warning';\n      case 'high':\n        return 'danger';\n      default:\n        return 'default';\n    }\n  };\n  this.loadExceptions = function() {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(flinkConfig.jobServer + \"jobs/\" + currentJob.jid + \"/exceptions\").success(function(exceptions) {\n          currentJob.exceptions = exceptions;\n          return deferred.resolve(exceptions);\n        });\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.cancelJob = function(jobid) {\n    return $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/yarn-cancel\");\n  };\n  this.stopJob = function(jobid) {\n    return $http.get(\"jobs/\" + jobid + \"/yarn-stop\");\n  };\n  return this;\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n# ----------------------------------------------\r\n\r\n.directive 'metricsGraph', ->\r\n  template: '<div class=\"panel panel-default panel-metric\">\r\n               <div class=\"panel-heading\">\r\n                 <span class=\"metric-title\">{{metric.id}}</span>\r\n                 <div class=\"buttons\">\r\n                   <div class=\"btn-group\">\r\n                     <button type=\"button\" ng-class=\"[btnClasses, {active: metric.size != \\'big\\'}]\" ng-click=\"setSize(\\'small\\')\">Small</button>\r\n                     <button type=\"button\" ng-class=\"[btnClasses, {active: metric.size == \\'big\\'}]\" ng-click=\"setSize(\\'big\\')\">Big</button>\r\n                   </div>\r\n                   <a title=\"Remove\" class=\"btn btn-default btn-xs remove\" ng-click=\"removeMetric()\"><i class=\"fa fa-close\" /></a>\r\n                 </div>\r\n               </div>\r\n               <div class=\"panel-body\">\r\n                  <svg />\r\n               </div>\r\n             </div>'\r\n  replace: true\r\n  scope:\r\n    metric: \"=\"\r\n    window: \"=\"\r\n    removeMetric: \"&\"\r\n    setMetricSize: \"=\"\r\n    getValues: \"&\"\r\n\r\n  link: (scope, element, attrs) ->\r\n    scope.btnClasses = ['btn', 'btn-default', 'btn-xs']\r\n\r\n    scope.value = null\r\n    scope.data = [{\r\n      values: scope.getValues()\r\n    }]\r\n\r\n    scope.options = {\r\n      x: (d, i) ->\r\n        d.x\r\n      y: (d, i) ->\r\n        d.y\r\n\r\n      xTickFormat: (d) ->\r\n        d3.time.format('%H:%M:%S')(new Date(d))\r\n\r\n      yTickFormat: (d) ->\r\n        found = false\r\n        pow = 0\r\n        step = 1\r\n        absD = Math.abs(d)\r\n\r\n        while !found && pow < 50\r\n          if Math.pow(10, pow) <= absD && absD < Math.pow(10, pow + step)\r\n            found = true\r\n          else\r\n            pow += step\r\n\r\n        if found && pow > 6\r\n          \"#{d / Math.pow(10, pow)}E#{pow}\"\r\n        else\r\n          \"#{d}\"\r\n    }\r\n\r\n    scope.showChart = ->\r\n      d3.select(element.find(\"svg\")[0])\r\n      .datum(scope.data)\r\n      .transition().duration(250)\r\n      .call(scope.chart)\r\n\r\n    scope.chart = nv.models.lineChart()\r\n      .options(scope.options)\r\n      .showLegend(false)\r\n      .margin({\r\n        top: 15\r\n        left: 60\r\n        bottom: 30\r\n        right: 30\r\n      })\r\n\r\n    scope.chart.yAxis.showMaxMin(false)\r\n    scope.chart.tooltip.hideDelay(0)\r\n    scope.chart.tooltip.contentGenerator((obj) ->\r\n      \"<p>#{d3.time.format('%H:%M:%S')(new Date(obj.point.x))} | #{obj.point.y}</p>\"\r\n    )\r\n\r\n    nv.utils.windowResize(scope.chart.update);\r\n\r\n    scope.setSize = (size) ->\r\n      scope.setMetricSize(scope.metric, size)\r\n\r\n    scope.showChart()\r\n\r\n    scope.$on 'metrics:data:update', (event, timestamp, data) ->\r\n#      scope.value = parseInt(data[scope.metric.id])\r\n      scope.value = parseFloat(data[scope.metric.id])\r\n\r\n      scope.data[0].values.push {\r\n        x: timestamp\r\n        y: scope.value\r\n      }\r\n\r\n      if scope.data[0].values.length > scope.window\r\n        scope.data[0].values.shift()\r\n\r\n      scope.showChart()\r\n      scope.chart.clearHighlights()\r\n      scope.chart.tooltip.hidden(true)\r\n\r\n    element.find(\".metric-title\").qtip({\r\n      content: {\r\n        text: scope.metric.id\r\n      },\r\n      position: {\r\n        my: 'bottom left',\r\n        at: 'top left'\r\n      },\r\n      style: {\r\n        classes: 'qtip-light qtip-timeline-bar'\r\n      }\r\n    });\r\n","angular.module('flinkApp').directive('metricsGraph', function() {\n  return {\n    template: '<div class=\"panel panel-default panel-metric\"> <div class=\"panel-heading\"> <span class=\"metric-title\">{{metric.id}}</span> <div class=\"buttons\"> <div class=\"btn-group\"> <button type=\"button\" ng-class=\"[btnClasses, {active: metric.size != \\'big\\'}]\" ng-click=\"setSize(\\'small\\')\">Small</button> <button type=\"button\" ng-class=\"[btnClasses, {active: metric.size == \\'big\\'}]\" ng-click=\"setSize(\\'big\\')\">Big</button> </div> <a title=\"Remove\" class=\"btn btn-default btn-xs remove\" ng-click=\"removeMetric()\"><i class=\"fa fa-close\" /></a> </div> </div> <div class=\"panel-body\"> <svg /> </div> </div>',\n    replace: true,\n    scope: {\n      metric: \"=\",\n      window: \"=\",\n      removeMetric: \"&\",\n      setMetricSize: \"=\",\n      getValues: \"&\"\n    },\n    link: function(scope, element, attrs) {\n      scope.btnClasses = ['btn', 'btn-default', 'btn-xs'];\n      scope.value = null;\n      scope.data = [\n        {\n          values: scope.getValues()\n        }\n      ];\n      scope.options = {\n        x: function(d, i) {\n          return d.x;\n        },\n        y: function(d, i) {\n          return d.y;\n        },\n        xTickFormat: function(d) {\n          return d3.time.format('%H:%M:%S')(new Date(d));\n        },\n        yTickFormat: function(d) {\n          var absD, found, pow, step;\n          found = false;\n          pow = 0;\n          step = 1;\n          absD = Math.abs(d);\n          while (!found && pow < 50) {\n            if (Math.pow(10, pow) <= absD && absD < Math.pow(10, pow + step)) {\n              found = true;\n            } else {\n              pow += step;\n            }\n          }\n          if (found && pow > 6) {\n            return (d / Math.pow(10, pow)) + \"E\" + pow;\n          } else {\n            return \"\" + d;\n          }\n        }\n      };\n      scope.showChart = function() {\n        return d3.select(element.find(\"svg\")[0]).datum(scope.data).transition().duration(250).call(scope.chart);\n      };\n      scope.chart = nv.models.lineChart().options(scope.options).showLegend(false).margin({\n        top: 15,\n        left: 60,\n        bottom: 30,\n        right: 30\n      });\n      scope.chart.yAxis.showMaxMin(false);\n      scope.chart.tooltip.hideDelay(0);\n      scope.chart.tooltip.contentGenerator(function(obj) {\n        return \"<p>\" + (d3.time.format('%H:%M:%S')(new Date(obj.point.x))) + \" | \" + obj.point.y + \"</p>\";\n      });\n      nv.utils.windowResize(scope.chart.update);\n      scope.setSize = function(size) {\n        return scope.setMetricSize(scope.metric, size);\n      };\n      scope.showChart();\n      scope.$on('metrics:data:update', function(event, timestamp, data) {\n        scope.value = parseFloat(data[scope.metric.id]);\n        scope.data[0].values.push({\n          x: timestamp,\n          y: scope.value\n        });\n        if (scope.data[0].values.length > scope.window) {\n          scope.data[0].values.shift();\n        }\n        scope.showChart();\n        scope.chart.clearHighlights();\n        return scope.chart.tooltip.hidden(true);\n      });\n      return element.find(\".metric-title\").qtip({\n        content: {\n          text: scope.metric.id\n        },\n        position: {\n          my: 'bottom left',\n          at: 'top left'\n        },\n        style: {\n          classes: 'qtip-light qtip-timeline-bar'\n        }\n      });\n    }\n  };\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.service 'MetricsService', ($http, $q, flinkConfig, $interval) ->\r\n  @metrics = {}\r\n  @values = {}\r\n  @watched = {}\r\n  @observer = {\r\n    jobid: null\r\n    nodeid: null\r\n    callback: null\r\n  }\r\n\r\n  @refresh = $interval =>\r\n    angular.forEach @metrics, (vertices, jobid) =>\r\n      angular.forEach vertices, (metrics, nodeid) =>\r\n        names = []\r\n        angular.forEach metrics, (metric, index) =>\r\n          names.push metric.id\r\n\r\n        if names.length > 0\r\n          @getMetrics(jobid, nodeid, names).then (values) =>\r\n            if jobid == @observer.jobid && nodeid == @observer.nodeid\r\n              @observer.callback(values) if @observer.callback\r\n\r\n\r\n  , flinkConfig[\"refresh-interval\"]\r\n\r\n  @registerObserver = (jobid, nodeid, callback) ->\r\n    @observer.jobid = jobid\r\n    @observer.nodeid = nodeid\r\n    @observer.callback = callback\r\n\r\n  @unRegisterObserver = ->\r\n    @observer = {\r\n      jobid: null\r\n      nodeid: null\r\n      callback: null\r\n    }\r\n\r\n  @setupMetrics = (jobid, vertices) ->\r\n    @setupLS()\r\n\r\n    @watched[jobid] = []\r\n    angular.forEach vertices, (v, k) =>\r\n      @watched[jobid].push(v.id) if v.id\r\n\r\n  @getWindow = ->\r\n    100\r\n\r\n  @setupLS = ->\r\n    if !localStorage.flinkMetrics?\r\n      @saveSetup()\r\n\r\n    @metrics = JSON.parse(localStorage.flinkMetrics)\r\n\r\n  @saveSetup = ->\r\n    localStorage.flinkMetrics = JSON.stringify(@metrics)\r\n\r\n  @saveValue = (jobid, nodeid, value) ->\r\n    unless @values[jobid]?\r\n      @values[jobid] = {}\r\n\r\n    unless @values[jobid][nodeid]?\r\n      @values[jobid][nodeid] = []\r\n\r\n    @values[jobid][nodeid].push(value)\r\n\r\n    if @values[jobid][nodeid].length > @getWindow()\r\n      @values[jobid][nodeid].shift()\r\n\r\n  @getValues = (jobid, nodeid, metricid) ->\r\n    return [] unless @values[jobid]?\r\n    return [] unless @values[jobid][nodeid]?\r\n\r\n    results = []\r\n    angular.forEach @values[jobid][nodeid], (v, k) =>\r\n      if v.values[metricid]?\r\n        results.push {\r\n          x: v.timestamp\r\n          y: v.values[metricid]\r\n        }\r\n\r\n    results\r\n\r\n  @setupLSFor = (jobid, nodeid) ->\r\n    if !@metrics[jobid]?\r\n      @metrics[jobid] = {}\r\n\r\n    if !@metrics[jobid][nodeid]?\r\n      @metrics[jobid][nodeid] = []\r\n\r\n  @addMetric = (jobid, nodeid, metricid) ->\r\n    @setupLSFor(jobid, nodeid)\r\n\r\n    @metrics[jobid][nodeid].push({id: metricid, size: 'small'})\r\n\r\n    @saveSetup()\r\n\r\n  @removeMetric = (jobid, nodeid, metric) =>\r\n    if @metrics[jobid][nodeid]?\r\n      i = @metrics[jobid][nodeid].indexOf(metric)\r\n      i = _.findIndex(@metrics[jobid][nodeid], { id: metric }) if i == -1\r\n\r\n      @metrics[jobid][nodeid].splice(i, 1) if i != -1\r\n\r\n      @saveSetup()\r\n\r\n  @setMetricSize = (jobid, nodeid, metric, size) =>\r\n    if @metrics[jobid][nodeid]?\r\n      i = @metrics[jobid][nodeid].indexOf(metric.id)\r\n      i = _.findIndex(@metrics[jobid][nodeid], { id: metric.id }) if i == -1\r\n\r\n      @metrics[jobid][nodeid][i] = { id: metric.id, size: size } if i != -1\r\n\r\n      @saveSetup()\r\n\r\n  @orderMetrics = (jobid, nodeid, item, index) ->\r\n    @setupLSFor(jobid, nodeid)\r\n\r\n    angular.forEach @metrics[jobid][nodeid], (v, k) =>\r\n      if v.id == item.id\r\n        @metrics[jobid][nodeid].splice(k, 1)\r\n        if k < index\r\n          index = index - 1\r\n\r\n    @metrics[jobid][nodeid].splice(index, 0, item)\r\n\r\n    @saveSetup()\r\n\r\n  @getMetricsSetup = (jobid, nodeid) =>\r\n    {\r\n      names: _.map(@metrics[jobid][nodeid], (value) =>\r\n        if _.isString(value) then { id: value, size: \"small\" } else value\r\n      )\r\n    }\r\n\r\n  @getAvailableMetrics = (jobid, nodeid) =>\r\n    @setupLSFor(jobid, nodeid)\r\n\r\n    deferred = $q.defer()\r\n\r\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics\"\r\n    .success (data) =>\r\n      results = []\r\n      angular.forEach data, (v, k) =>\r\n        i = @metrics[jobid][nodeid].indexOf(v.id)\r\n        i = _.findIndex(@metrics[jobid][nodeid], { id: v.id }) if i == -1\r\n\r\n        if i == -1\r\n          results.push(v)\r\n\r\n      deferred.resolve(results)\r\n\r\n    deferred.promise\r\n\r\n  @getAllAvailableMetrics = (jobid, nodeid) =>\r\n    deferred = $q.defer()\r\n\r\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics\"\r\n    .success (data) =>\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @getMetrics = (jobid, nodeid, metricIds) ->\r\n    deferred = $q.defer()\r\n\r\n    ids = metricIds.join(\",\")\r\n\r\n    $http.get flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics?get=\" + ids\r\n    .success (data) =>\r\n      result = {}\r\n      angular.forEach data, (v, k) ->\r\n        result[v.id] = parseInt(v.value)\r\n\r\n      newValue = {\r\n        timestamp: Date.now()\r\n        values: result\r\n      }\r\n      @saveValue(jobid, nodeid, newValue)\r\n      deferred.resolve(newValue)\r\n\r\n    deferred.promise\r\n\r\n  @setupLS()\r\n\r\n  @\r\n","angular.module('flinkApp').service('MetricsService', function($http, $q, flinkConfig, $interval) {\n  this.metrics = {};\n  this.values = {};\n  this.watched = {};\n  this.observer = {\n    jobid: null,\n    nodeid: null,\n    callback: null\n  };\n  this.refresh = $interval((function(_this) {\n    return function() {\n      return angular.forEach(_this.metrics, function(vertices, jobid) {\n        return angular.forEach(vertices, function(metrics, nodeid) {\n          var names;\n          names = [];\n          angular.forEach(metrics, function(metric, index) {\n            return names.push(metric.id);\n          });\n          if (names.length > 0) {\n            return _this.getMetrics(jobid, nodeid, names).then(function(values) {\n              if (jobid === _this.observer.jobid && nodeid === _this.observer.nodeid) {\n                if (_this.observer.callback) {\n                  return _this.observer.callback(values);\n                }\n              }\n            });\n          }\n        });\n      });\n    };\n  })(this), flinkConfig[\"refresh-interval\"]);\n  this.registerObserver = function(jobid, nodeid, callback) {\n    this.observer.jobid = jobid;\n    this.observer.nodeid = nodeid;\n    return this.observer.callback = callback;\n  };\n  this.unRegisterObserver = function() {\n    return this.observer = {\n      jobid: null,\n      nodeid: null,\n      callback: null\n    };\n  };\n  this.setupMetrics = function(jobid, vertices) {\n    this.setupLS();\n    this.watched[jobid] = [];\n    return angular.forEach(vertices, (function(_this) {\n      return function(v, k) {\n        if (v.id) {\n          return _this.watched[jobid].push(v.id);\n        }\n      };\n    })(this));\n  };\n  this.getWindow = function() {\n    return 100;\n  };\n  this.setupLS = function() {\n    if (localStorage.flinkMetrics == null) {\n      this.saveSetup();\n    }\n    return this.metrics = JSON.parse(localStorage.flinkMetrics);\n  };\n  this.saveSetup = function() {\n    return localStorage.flinkMetrics = JSON.stringify(this.metrics);\n  };\n  this.saveValue = function(jobid, nodeid, value) {\n    if (this.values[jobid] == null) {\n      this.values[jobid] = {};\n    }\n    if (this.values[jobid][nodeid] == null) {\n      this.values[jobid][nodeid] = [];\n    }\n    this.values[jobid][nodeid].push(value);\n    if (this.values[jobid][nodeid].length > this.getWindow()) {\n      return this.values[jobid][nodeid].shift();\n    }\n  };\n  this.getValues = function(jobid, nodeid, metricid) {\n    var results;\n    if (this.values[jobid] == null) {\n      return [];\n    }\n    if (this.values[jobid][nodeid] == null) {\n      return [];\n    }\n    results = [];\n    angular.forEach(this.values[jobid][nodeid], (function(_this) {\n      return function(v, k) {\n        if (v.values[metricid] != null) {\n          return results.push({\n            x: v.timestamp,\n            y: v.values[metricid]\n          });\n        }\n      };\n    })(this));\n    return results;\n  };\n  this.setupLSFor = function(jobid, nodeid) {\n    if (this.metrics[jobid] == null) {\n      this.metrics[jobid] = {};\n    }\n    if (this.metrics[jobid][nodeid] == null) {\n      return this.metrics[jobid][nodeid] = [];\n    }\n  };\n  this.addMetric = function(jobid, nodeid, metricid) {\n    this.setupLSFor(jobid, nodeid);\n    this.metrics[jobid][nodeid].push({\n      id: metricid,\n      size: 'small'\n    });\n    return this.saveSetup();\n  };\n  this.removeMetric = (function(_this) {\n    return function(jobid, nodeid, metric) {\n      var i;\n      if (_this.metrics[jobid][nodeid] != null) {\n        i = _this.metrics[jobid][nodeid].indexOf(metric);\n        if (i === -1) {\n          i = _.findIndex(_this.metrics[jobid][nodeid], {\n            id: metric\n          });\n        }\n        if (i !== -1) {\n          _this.metrics[jobid][nodeid].splice(i, 1);\n        }\n        return _this.saveSetup();\n      }\n    };\n  })(this);\n  this.setMetricSize = (function(_this) {\n    return function(jobid, nodeid, metric, size) {\n      var i;\n      if (_this.metrics[jobid][nodeid] != null) {\n        i = _this.metrics[jobid][nodeid].indexOf(metric.id);\n        if (i === -1) {\n          i = _.findIndex(_this.metrics[jobid][nodeid], {\n            id: metric.id\n          });\n        }\n        if (i !== -1) {\n          _this.metrics[jobid][nodeid][i] = {\n            id: metric.id,\n            size: size\n          };\n        }\n        return _this.saveSetup();\n      }\n    };\n  })(this);\n  this.orderMetrics = function(jobid, nodeid, item, index) {\n    this.setupLSFor(jobid, nodeid);\n    angular.forEach(this.metrics[jobid][nodeid], (function(_this) {\n      return function(v, k) {\n        if (v.id === item.id) {\n          _this.metrics[jobid][nodeid].splice(k, 1);\n          if (k < index) {\n            return index = index - 1;\n          }\n        }\n      };\n    })(this));\n    this.metrics[jobid][nodeid].splice(index, 0, item);\n    return this.saveSetup();\n  };\n  this.getMetricsSetup = (function(_this) {\n    return function(jobid, nodeid) {\n      return {\n        names: _.map(_this.metrics[jobid][nodeid], function(value) {\n          if (_.isString(value)) {\n            return {\n              id: value,\n              size: \"small\"\n            };\n          } else {\n            return value;\n          }\n        })\n      };\n    };\n  })(this);\n  this.getAvailableMetrics = (function(_this) {\n    return function(jobid, nodeid) {\n      var deferred;\n      _this.setupLSFor(jobid, nodeid);\n      deferred = $q.defer();\n      $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics\").success(function(data) {\n        var results;\n        results = [];\n        angular.forEach(data, function(v, k) {\n          var i;\n          i = _this.metrics[jobid][nodeid].indexOf(v.id);\n          if (i === -1) {\n            i = _.findIndex(_this.metrics[jobid][nodeid], {\n              id: v.id\n            });\n          }\n          if (i === -1) {\n            return results.push(v);\n          }\n        });\n        return deferred.resolve(results);\n      });\n      return deferred.promise;\n    };\n  })(this);\n  this.getAllAvailableMetrics = (function(_this) {\n    return function(jobid, nodeid) {\n      var deferred;\n      deferred = $q.defer();\n      $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics\").success(function(data) {\n        return deferred.resolve(data);\n      });\n      return deferred.promise;\n    };\n  })(this);\n  this.getMetrics = function(jobid, nodeid, metricIds) {\n    var deferred, ids;\n    deferred = $q.defer();\n    ids = metricIds.join(\",\");\n    $http.get(flinkConfig.jobServer + \"jobs/\" + jobid + \"/vertices/\" + nodeid + \"/metrics?get=\" + ids).success((function(_this) {\n      return function(data) {\n        var newValue, result;\n        result = {};\n        angular.forEach(data, function(v, k) {\n          return result[v.id] = parseInt(v.value);\n        });\n        newValue = {\n          timestamp: Date.now(),\n          values: result\n        };\n        _this.saveValue(jobid, nodeid, newValue);\n        return deferred.resolve(newValue);\n      };\n    })(this));\n    return deferred.promise;\n  };\n  this.setupLS();\n  return this;\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.controller 'OverviewController', ($scope, OverviewService, JobsService, $interval, flinkConfig) ->\r\n  $scope.jobObserver = ->\r\n    $scope.runningJobs = JobsService.getJobs('running')\r\n    $scope.finishedJobs = JobsService.getJobs('finished')\r\n\r\n  JobsService.registerObserver($scope.jobObserver)\r\n  $scope.$on '$destroy', ->\r\n    JobsService.unRegisterObserver($scope.jobObserver)\r\n\r\n  $scope.jobObserver()\r\n\r\n  OverviewService.loadOverview().then (data) ->\r\n    $scope.overview = data\r\n\r\n  refresh = $interval ->\r\n    OverviewService.loadOverview().then (data) ->\r\n      $scope.overview = data\r\n  , flinkConfig[\"refresh-interval\"]\r\n\r\n  $scope.$on '$destroy', ->\r\n    $interval.cancel(refresh)\r\n","angular.module('flinkApp').controller('OverviewController', function($scope, OverviewService, JobsService, $interval, flinkConfig) {\n  var refresh;\n  $scope.jobObserver = function() {\n    $scope.runningJobs = JobsService.getJobs('running');\n    return $scope.finishedJobs = JobsService.getJobs('finished');\n  };\n  JobsService.registerObserver($scope.jobObserver);\n  $scope.$on('$destroy', function() {\n    return JobsService.unRegisterObserver($scope.jobObserver);\n  });\n  $scope.jobObserver();\n  OverviewService.loadOverview().then(function(data) {\n    return $scope.overview = data;\n  });\n  refresh = $interval(function() {\n    return OverviewService.loadOverview().then(function(data) {\n      return $scope.overview = data;\n    });\n  }, flinkConfig[\"refresh-interval\"]);\n  return $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.service 'OverviewService', ($http, flinkConfig, $q) ->\r\n  overview = {}\r\n\r\n  @loadOverview = ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"overview\")\r\n    .success (data, status, headers, config) ->\r\n      overview = data\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @\r\n","angular.module('flinkApp').service('OverviewService', function($http, flinkConfig, $q) {\n  var overview;\n  overview = {};\n  this.loadOverview = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"overview\").success(function(data, status, headers, config) {\n      overview = data;\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.controller 'JobSubmitController', ($scope, JobSubmitService, $interval, flinkConfig, $state, $location) ->\r\n  $scope.yarn = $location.absUrl().indexOf(\"/proxy/application_\") != -1\r\n  $scope.loadList = () ->\r\n    JobSubmitService.loadJarList().then (data) ->\r\n      $scope.address = data.address\r\n      $scope.noaccess = data.error\r\n      $scope.jars = data.files\r\n\r\n  $scope.defaultState = () ->\r\n    $scope.plan = null\r\n    $scope.error = null\r\n    $scope.state = {\r\n      selected: null,\r\n      parallelism: \"\",\r\n      savepointPath: \"\",\r\n      allowNonRestoredState: false\r\n      'entry-class': \"\",\r\n      'program-args': \"\",\r\n      'plan-button': \"Show Plan\",\r\n      'submit-button': \"Submit\",\r\n      'action-time': 0\r\n    }\r\n\r\n  $scope.defaultState()\r\n  $scope.uploader = {}\r\n  $scope.loadList()\r\n\r\n  refresh = $interval ->\r\n    $scope.loadList()\r\n  , flinkConfig[\"refresh-interval\"]\r\n\r\n  $scope.$on '$destroy', ->\r\n    $interval.cancel(refresh)\r\n\r\n  $scope.selectJar = (id) ->\r\n    if $scope.state.selected == id\r\n      $scope.defaultState()\r\n    else\r\n      $scope.defaultState()\r\n      $scope.state.selected = id\r\n\r\n  $scope.deleteJar = (event, id) ->\r\n    if $scope.state.selected == id\r\n      $scope.defaultState()\r\n    angular.element(event.currentTarget).removeClass(\"fa-remove\").addClass(\"fa-spin fa-spinner\")\r\n    JobSubmitService.deleteJar(id).then (data) ->\r\n      angular.element(event.currentTarget).removeClass(\"fa-spin fa-spinner\").addClass(\"fa-remove\")\r\n      if data.error?\r\n        alert(data.error)\r\n\r\n  $scope.loadEntryClass = (name) ->\r\n    $scope.state['entry-class'] = name\r\n\r\n  $scope.getPlan = () ->\r\n    if $scope.state['plan-button'] == \"Show Plan\"\r\n      action = new Date().getTime()\r\n      $scope.state['action-time'] = action\r\n      $scope.state['submit-button'] = \"Submit\"\r\n      $scope.state['plan-button'] = \"Getting Plan\"\r\n      $scope.error = null\r\n      $scope.plan = null\r\n      JobSubmitService.getPlan(\r\n        $scope.state.selected, {\r\n          'entry-class': $scope.state['entry-class'],\r\n          parallelism: $scope.state.parallelism,\r\n          'program-args': $scope.state['program-args']\r\n        }\r\n      ).then (data) ->\r\n        if action == $scope.state['action-time']\r\n          $scope.state['plan-button'] = \"Show Plan\"\r\n          $scope.error = data.error\r\n          $scope.plan = data.plan\r\n\r\n  $scope.runJob = () ->\r\n    if $scope.state['submit-button'] == \"Submit\"\r\n      action = new Date().getTime()\r\n      $scope.state['action-time'] = action\r\n      $scope.state['submit-button'] = \"Submitting\"\r\n      $scope.state['plan-button'] = \"Show Plan\"\r\n      $scope.error = null\r\n      JobSubmitService.runJob(\r\n        $scope.state.selected, {\r\n          'entry-class': $scope.state['entry-class'],\r\n          parallelism: $scope.state.parallelism,\r\n          'program-args': $scope.state['program-args'],\r\n          savepointPath: $scope.state['savepointPath'],\r\n          allowNonRestoredState: $scope.state['allowNonRestoredState']\r\n        }\r\n      ).then (data) ->\r\n        if action == $scope.state['action-time']\r\n          $scope.state['submit-button'] = \"Submit\"\r\n          $scope.error = data.error\r\n          if data.jobid?\r\n            $state.go(\"single-job.plan.subtasks\", {jobid: data.jobid})\r\n\r\n  # job plan display related stuff\r\n  $scope.nodeid = null\r\n  $scope.changeNode = (nodeid) ->\r\n    if nodeid != $scope.nodeid\r\n      $scope.nodeid = nodeid\r\n      $scope.vertex = null\r\n      $scope.subtasks = null\r\n      $scope.accumulators = null\r\n\r\n      $scope.$broadcast 'reload'\r\n\r\n    else\r\n      $scope.nodeid = null\r\n      $scope.nodeUnfolded = false\r\n      $scope.vertex = null\r\n      $scope.subtasks = null\r\n      $scope.accumulators = null\r\n\r\n  $scope.clearFiles = () ->\r\n    $scope.uploader = {}\r\n\r\n  $scope.uploadFiles = (files) ->\r\n    # make sure everything is clear again.\r\n    $scope.uploader = {}\r\n    if files.length == 1\r\n      $scope.uploader['file'] = files[0]\r\n      $scope.uploader['upload'] = true\r\n    else\r\n      $scope.uploader['error'] = \"Did ya forget to select a file?\"\r\n\r\n  $scope.startUpload = () ->\r\n    if $scope.uploader['file']?\r\n      formdata = new FormData()\r\n      formdata.append(\"jarfile\", $scope.uploader['file'])\r\n      $scope.uploader['upload'] = false\r\n      $scope.uploader['success'] = \"Initializing upload...\"\r\n      xhr = new XMLHttpRequest()\r\n      xhr.upload.onprogress = (event) ->\r\n        $scope.uploader['success'] = null\r\n        $scope.uploader['progress'] = parseInt(100 * event.loaded / event.total)\r\n      xhr.upload.onerror = (event) ->\r\n        $scope.uploader['progress'] = null\r\n        $scope.uploader['error'] = \"An error occurred while uploading your file\"\r\n      xhr.upload.onload = (event) ->\r\n        $scope.uploader['progress'] = null\r\n        $scope.uploader['success'] = \"Saving...\"\r\n      xhr.onreadystatechange = () ->\r\n        if xhr.readyState == 4\r\n          response = JSON.parse(xhr.responseText)\r\n          if response.error?\r\n            $scope.uploader['error'] = response.error\r\n            $scope.uploader['success'] = null\r\n          else\r\n            $scope.uploader['success'] = \"Uploaded!\"\r\n      xhr.open(\"POST\", \"/jars/upload\")\r\n      xhr.send(formdata)\r\n    else\r\n      console.log(\"Unexpected Error. This should not happen\")\r\n\r\n.filter 'getJarSelectClass', ->\r\n  (selected, actual) ->\r\n    if selected == actual\r\n      \"fa-check-square\"\r\n    else\r\n      \"fa-square-o\"\r\n","angular.module('flinkApp').controller('JobSubmitController', function($scope, JobSubmitService, $interval, flinkConfig, $state, $location) {\n  var refresh;\n  $scope.yarn = $location.absUrl().indexOf(\"/proxy/application_\") !== -1;\n  $scope.loadList = function() {\n    return JobSubmitService.loadJarList().then(function(data) {\n      $scope.address = data.address;\n      $scope.noaccess = data.error;\n      return $scope.jars = data.files;\n    });\n  };\n  $scope.defaultState = function() {\n    $scope.plan = null;\n    $scope.error = null;\n    return $scope.state = {\n      selected: null,\n      parallelism: \"\",\n      savepointPath: \"\",\n      allowNonRestoredState: false,\n      'entry-class': \"\",\n      'program-args': \"\",\n      'plan-button': \"Show Plan\",\n      'submit-button': \"Submit\",\n      'action-time': 0\n    };\n  };\n  $scope.defaultState();\n  $scope.uploader = {};\n  $scope.loadList();\n  refresh = $interval(function() {\n    return $scope.loadList();\n  }, flinkConfig[\"refresh-interval\"]);\n  $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n  $scope.selectJar = function(id) {\n    if ($scope.state.selected === id) {\n      return $scope.defaultState();\n    } else {\n      $scope.defaultState();\n      return $scope.state.selected = id;\n    }\n  };\n  $scope.deleteJar = function(event, id) {\n    if ($scope.state.selected === id) {\n      $scope.defaultState();\n    }\n    angular.element(event.currentTarget).removeClass(\"fa-remove\").addClass(\"fa-spin fa-spinner\");\n    return JobSubmitService.deleteJar(id).then(function(data) {\n      angular.element(event.currentTarget).removeClass(\"fa-spin fa-spinner\").addClass(\"fa-remove\");\n      if (data.error != null) {\n        return alert(data.error);\n      }\n    });\n  };\n  $scope.loadEntryClass = function(name) {\n    return $scope.state['entry-class'] = name;\n  };\n  $scope.getPlan = function() {\n    var action;\n    if ($scope.state['plan-button'] === \"Show Plan\") {\n      action = new Date().getTime();\n      $scope.state['action-time'] = action;\n      $scope.state['submit-button'] = \"Submit\";\n      $scope.state['plan-button'] = \"Getting Plan\";\n      $scope.error = null;\n      $scope.plan = null;\n      return JobSubmitService.getPlan($scope.state.selected, {\n        'entry-class': $scope.state['entry-class'],\n        parallelism: $scope.state.parallelism,\n        'program-args': $scope.state['program-args']\n      }).then(function(data) {\n        if (action === $scope.state['action-time']) {\n          $scope.state['plan-button'] = \"Show Plan\";\n          $scope.error = data.error;\n          return $scope.plan = data.plan;\n        }\n      });\n    }\n  };\n  $scope.runJob = function() {\n    var action;\n    if ($scope.state['submit-button'] === \"Submit\") {\n      action = new Date().getTime();\n      $scope.state['action-time'] = action;\n      $scope.state['submit-button'] = \"Submitting\";\n      $scope.state['plan-button'] = \"Show Plan\";\n      $scope.error = null;\n      return JobSubmitService.runJob($scope.state.selected, {\n        'entry-class': $scope.state['entry-class'],\n        parallelism: $scope.state.parallelism,\n        'program-args': $scope.state['program-args'],\n        savepointPath: $scope.state['savepointPath'],\n        allowNonRestoredState: $scope.state['allowNonRestoredState']\n      }).then(function(data) {\n        if (action === $scope.state['action-time']) {\n          $scope.state['submit-button'] = \"Submit\";\n          $scope.error = data.error;\n          if (data.jobid != null) {\n            return $state.go(\"single-job.plan.subtasks\", {\n              jobid: data.jobid\n            });\n          }\n        }\n      });\n    }\n  };\n  $scope.nodeid = null;\n  $scope.changeNode = function(nodeid) {\n    if (nodeid !== $scope.nodeid) {\n      $scope.nodeid = nodeid;\n      $scope.vertex = null;\n      $scope.subtasks = null;\n      $scope.accumulators = null;\n      return $scope.$broadcast('reload');\n    } else {\n      $scope.nodeid = null;\n      $scope.nodeUnfolded = false;\n      $scope.vertex = null;\n      $scope.subtasks = null;\n      return $scope.accumulators = null;\n    }\n  };\n  $scope.clearFiles = function() {\n    return $scope.uploader = {};\n  };\n  $scope.uploadFiles = function(files) {\n    $scope.uploader = {};\n    if (files.length === 1) {\n      $scope.uploader['file'] = files[0];\n      return $scope.uploader['upload'] = true;\n    } else {\n      return $scope.uploader['error'] = \"Did ya forget to select a file?\";\n    }\n  };\n  return $scope.startUpload = function() {\n    var formdata, xhr;\n    if ($scope.uploader['file'] != null) {\n      formdata = new FormData();\n      formdata.append(\"jarfile\", $scope.uploader['file']);\n      $scope.uploader['upload'] = false;\n      $scope.uploader['success'] = \"Initializing upload...\";\n      xhr = new XMLHttpRequest();\n      xhr.upload.onprogress = function(event) {\n        $scope.uploader['success'] = null;\n        return $scope.uploader['progress'] = parseInt(100 * event.loaded / event.total);\n      };\n      xhr.upload.onerror = function(event) {\n        $scope.uploader['progress'] = null;\n        return $scope.uploader['error'] = \"An error occurred while uploading your file\";\n      };\n      xhr.upload.onload = function(event) {\n        $scope.uploader['progress'] = null;\n        return $scope.uploader['success'] = \"Saving...\";\n      };\n      xhr.onreadystatechange = function() {\n        var response;\n        if (xhr.readyState === 4) {\n          response = JSON.parse(xhr.responseText);\n          if (response.error != null) {\n            $scope.uploader['error'] = response.error;\n            return $scope.uploader['success'] = null;\n          } else {\n            return $scope.uploader['success'] = \"Uploaded!\";\n          }\n        }\n      };\n      xhr.open(\"POST\", \"/jars/upload\");\n      return xhr.send(formdata);\n    } else {\n      return console.log(\"Unexpected Error. This should not happen\");\n    }\n  };\n}).filter('getJarSelectClass', function() {\n  return function(selected, actual) {\n    if (selected === actual) {\n      return \"fa-check-square\";\n    } else {\n      return \"fa-square-o\";\n    }\n  };\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.service 'JobSubmitService', ($http, flinkConfig, $q) ->\r\n\r\n  @loadJarList = () ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"jars/\")\r\n    .success (data, status, headers, config) ->\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @deleteJar = (id) ->\r\n    deferred = $q.defer()\r\n\r\n    $http.delete(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id))\r\n    .success (data, status, headers, config) ->\r\n       deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @getPlan = (id, args) ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id) + \"/plan\", {params: args})\r\n    .success (data, status, headers, config) ->\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @runJob = (id, args) ->\r\n    deferred = $q.defer()\r\n\r\n    $http.post(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id) + \"/run\", {}, {params: args})\r\n    .success (data, status, headers, config) ->\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @\r\n","angular.module('flinkApp').service('JobSubmitService', function($http, flinkConfig, $q) {\n  this.loadJarList = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jars/\").success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  this.deleteJar = function(id) {\n    var deferred;\n    deferred = $q.defer();\n    $http[\"delete\"](flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id)).success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  this.getPlan = function(id, args) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id) + \"/plan\", {\n      params: args\n    }).success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  this.runJob = function(id, args) {\n    var deferred;\n    deferred = $q.defer();\n    $http.post(flinkConfig.jobServer + \"jars/\" + encodeURIComponent(id) + \"/run\", {}, {\n      params: args\n    }).success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.controller 'AllTaskManagersController', ($scope, TaskManagersService, $interval, flinkConfig) ->\r\n  TaskManagersService.loadManagers().then (data) ->\r\n    $scope.managers = data\r\n\r\n  refresh = $interval ->\r\n    TaskManagersService.loadManagers().then (data) ->\r\n      $scope.managers = data\r\n  , flinkConfig[\"refresh-interval\"]\r\n\r\n  $scope.$on '$destroy', ->\r\n    $interval.cancel(refresh)\r\n\r\n.controller 'SingleTaskManagerController', ($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) ->\r\n  $scope.metrics = {}\r\n  SingleTaskManagerService.loadMetrics($stateParams.taskmanagerid).then (data) ->\r\n      $scope.metrics = data[0]\r\n\r\n    refresh = $interval ->\r\n      SingleTaskManagerService.loadMetrics($stateParams.taskmanagerid).then (data) ->\r\n        $scope.metrics = data[0]\r\n    , flinkConfig[\"refresh-interval\"]\r\n\r\n    $scope.$on '$destroy', ->\r\n      $interval.cancel(refresh)\r\n\r\n.controller 'SingleTaskManagerLogsController', ($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) ->\r\n  $scope.log = {}\r\n  SingleTaskManagerService.loadLogs($stateParams.taskmanagerid).then (data) ->\r\n    $scope.log = data\r\n\r\n  $scope.reloadData = () ->\r\n    SingleTaskManagerService.loadLogs($stateParams.taskmanagerid).then (data) ->\r\n      $scope.log = data\r\n\r\n  $scope.downloadData = () ->\r\n    window.location.href = \"/taskmanagers/\" + ($stateParams.taskmanagerid) + \"/log\"\r\n\r\n.controller 'SingleTaskManagerStdoutController', ($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) ->\r\n  $scope.stdout = {}\r\n  SingleTaskManagerService.loadStdout($stateParams.taskmanagerid).then (data) ->\r\n    $scope.stdout = data\r\n\r\n  $scope.reloadData = () ->\r\n    SingleTaskManagerService.loadStdout($stateParams.taskmanagerid).then (data) ->\r\n      $scope.stdout = data\r\n\r\n  $scope.downloadData = () ->\r\n    window.location.href = \"/taskmanagers/\" + ($stateParams.taskmanagerid) + \"/stdout\"\r\n","angular.module('flinkApp').controller('AllTaskManagersController', function($scope, TaskManagersService, $interval, flinkConfig) {\n  var refresh;\n  TaskManagersService.loadManagers().then(function(data) {\n    return $scope.managers = data;\n  });\n  refresh = $interval(function() {\n    return TaskManagersService.loadManagers().then(function(data) {\n      return $scope.managers = data;\n    });\n  }, flinkConfig[\"refresh-interval\"]);\n  return $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n}).controller('SingleTaskManagerController', function($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) {\n  var refresh;\n  $scope.metrics = {};\n  SingleTaskManagerService.loadMetrics($stateParams.taskmanagerid).then(function(data) {\n    return $scope.metrics = data[0];\n  });\n  refresh = $interval(function() {\n    return SingleTaskManagerService.loadMetrics($stateParams.taskmanagerid).then(function(data) {\n      return $scope.metrics = data[0];\n    });\n  }, flinkConfig[\"refresh-interval\"]);\n  return $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n}).controller('SingleTaskManagerLogsController', function($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) {\n  $scope.log = {};\n  SingleTaskManagerService.loadLogs($stateParams.taskmanagerid).then(function(data) {\n    return $scope.log = data;\n  });\n  $scope.reloadData = function() {\n    return SingleTaskManagerService.loadLogs($stateParams.taskmanagerid).then(function(data) {\n      return $scope.log = data;\n    });\n  };\n  return $scope.downloadData = function() {\n    return window.location.href = \"/taskmanagers/\" + $stateParams.taskmanagerid + \"/log\";\n  };\n}).controller('SingleTaskManagerStdoutController', function($scope, $stateParams, SingleTaskManagerService, $interval, flinkConfig) {\n  $scope.stdout = {};\n  SingleTaskManagerService.loadStdout($stateParams.taskmanagerid).then(function(data) {\n    return $scope.stdout = data;\n  });\n  $scope.reloadData = function() {\n    return SingleTaskManagerService.loadStdout($stateParams.taskmanagerid).then(function(data) {\n      return $scope.stdout = data;\n    });\n  };\n  return $scope.downloadData = function() {\n    return window.location.href = \"/taskmanagers/\" + $stateParams.taskmanagerid + \"/stdout\";\n  };\n});\n","#\r\n# Licensed to the Apache Software Foundation (ASF) under one\r\n# or more contributor license agreements.  See the NOTICE file\r\n# distributed with this work for additional information\r\n# regarding copyright ownership.  The ASF licenses this file\r\n# to you under the Apache License, Version 2.0 (the\r\n# \"License\"); you may not use this file except in compliance\r\n# with the License.  You may obtain a copy of the License at\r\n#\r\n#     http://www.apache.org/licenses/LICENSE-2.0\r\n#\r\n# Unless required by applicable law or agreed to in writing, software\r\n# distributed under the License is distributed on an \"AS IS\" BASIS,\r\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n# See the License for the specific language governing permissions and\r\n# limitations under the License.\r\n#\r\n\r\nangular.module('flinkApp')\r\n\r\n.service 'TaskManagersService', ($http, flinkConfig, $q) ->\r\n  @loadManagers = () ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"taskmanagers\")\r\n    .success (data, status, headers, config) ->\r\n      deferred.resolve(data['taskmanagers'])\r\n\r\n    deferred.promise\r\n\r\n  @\r\n\r\n.service 'SingleTaskManagerService', ($http, flinkConfig, $q) ->\r\n  @loadMetrics = (taskmanagerid) ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/metrics\")\r\n    .success (data, status, headers, config) ->\r\n      deferred.resolve(data['taskmanagers'])\r\n\r\n    deferred.promise\r\n\r\n  @loadLogs = (taskmanagerid) ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/log\")\r\n    .success (data, status, headers, config) ->\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @loadStdout = (taskmanagerid) ->\r\n    deferred = $q.defer()\r\n\r\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/stdout\")\r\n    .success (data, status, headers, config) ->\r\n      deferred.resolve(data)\r\n\r\n    deferred.promise\r\n\r\n  @\r\n\r\n","angular.module('flinkApp').service('TaskManagersService', function($http, flinkConfig, $q) {\n  this.loadManagers = function() {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"taskmanagers\").success(function(data, status, headers, config) {\n      return deferred.resolve(data['taskmanagers']);\n    });\n    return deferred.promise;\n  };\n  return this;\n}).service('SingleTaskManagerService', function($http, flinkConfig, $q) {\n  this.loadMetrics = function(taskmanagerid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/metrics\").success(function(data, status, headers, config) {\n      return deferred.resolve(data['taskmanagers']);\n    });\n    return deferred.promise;\n  };\n  this.loadLogs = function(taskmanagerid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/log\").success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  this.loadStdout = function(taskmanagerid) {\n    var deferred;\n    deferred = $q.defer();\n    $http.get(flinkConfig.jobServer + \"taskmanagers/\" + taskmanagerid + \"/stdout\").success(function(data, status, headers, config) {\n      return deferred.resolve(data);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n"],"sourceRoot":"/source/"}