diff --git a/flink-runtime-web/src/main/java/org/apache/flink/runtime/webmonitor/WebRuntimeMonitor.java b/flink-runtime-web/src/main/java/org/apache/flink/runtime/webmonitor/WebRuntimeMonitor.java index ec973c7fe1ca0e6c51b16e8c8d5ece1a61a80cc1..40d9f2d1e292c0ac718f2adcd9698cf99336b211 100644 --- a/flink-runtime-web/src/main/java/org/apache/flink/runtime/webmonitor/WebRuntimeMonitor.java +++ b/flink-runtime-web/src/main/java/org/apache/flink/runtime/webmonitor/WebRuntimeMonitor.java @@ -192,8 +192,6 @@ public class WebRuntimeMonitor implements WebMonitor { // job manager configuration, log and stdout .GET("/jobmanager/config", handler(new JobManagerConfigHandler(config))) - .GET("/jobmanager/log", new StaticFileServerHandler(logDir)) - .GET("/jobmanager/stdout", new StaticFileServerHandler(outDir)) // overview over jobs .GET("/joboverview", handler(new CurrentJobsOverviewHandler(DEFAULT_REQUEST_TIMEOUT, true, true))) @@ -222,6 +220,8 @@ public class WebRuntimeMonitor implements WebMonitor { .GET("/taskmanagers", handler(new TaskManagersHandler(DEFAULT_REQUEST_TIMEOUT))) .GET("/taskmanagers/:" + TaskManagersHandler.TASK_MANAGER_ID_KEY, handler(new TaskManagersHandler(DEFAULT_REQUEST_TIMEOUT))) + .GET("/jobmanager/log", new StaticFileServerHandler(retriever, jobManagerAddressPromise.future(), timeout, logDir)) + .GET("/jobmanager/stdout", new StaticFileServerHandler(retriever, jobManagerAddressPromise.future(), timeout, outDir)) // this handler serves all the static contents .GET("/:*", new StaticFileServerHandler(retriever, jobManagerAddressPromise.future(), timeout, webRootDir)); diff --git a/flink-runtime-web/src/main/java/org/apache/flink/runtime/webmonitor/files/StaticFileServerHandler.java b/flink-runtime-web/src/main/java/org/apache/flink/runtime/webmonitor/files/StaticFileServerHandler.java index d46a900d40e3bc3aa8b9ff93cb0108c8d1440a5b..02dd81e9b3a44614c58bdd723ae46200b333c3c3 100644 --- a/flink-runtime-web/src/main/java/org/apache/flink/runtime/webmonitor/files/StaticFileServerHandler.java +++ b/flink-runtime-web/src/main/java/org/apache/flink/runtime/webmonitor/files/StaticFileServerHandler.java @@ -58,10 +58,12 @@ import scala.concurrent.duration.FiniteDuration; import java.io.File; import java.io.FileNotFoundException; +import java.io.IOException; import java.io.InputStream; import java.io.FilenameFilter; import java.io.RandomAccessFile; import java.nio.file.Files; +import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; @@ -198,7 +200,7 @@ public class StaticFileServerHandler extends SimpleChannelInboundHandler * Response when running with leading JobManager. */ private void respondAsLeader(ChannelHandlerContext ctx, HttpRequest request, String requestPath) - throws ParseException, IOException { + throws IOException, ParseException { // convert to absolute path final File file = new File(rootPath, requestPath); diff --git a/flink-runtime-web/src/main/resources/web/js/index.js.orig b/flink-runtime-web/src/main/resources/web/js/index.js.orig deleted file mode 100644 index 018e7437cc1a53c1588bf0157cbdb988925b3da2..0000000000000000000000000000000000000000 --- a/flink-runtime-web/src/main/resources/web/js/index.js.orig +++ /dev/null @@ -1,1150 +0,0 @@ -angular.module('flinkApp', ['ui.router', 'angularMoment']).run(["$rootScope", function($rootScope) { - $rootScope.sidebarVisible = false; - return $rootScope.showSidebar = function() { - $rootScope.sidebarVisible = !$rootScope.sidebarVisible; - return $rootScope.sidebarClass = 'force-show'; - }; -}]).value('flinkConfig', { - "refresh-interval": 10000 -}).run(["JobsService", "MainService", "flinkConfig", "$interval", function(JobsService, MainService, flinkConfig, $interval) { - return MainService.loadConfig().then(function(config) { - angular.extend(flinkConfig, config); - JobsService.listJobs(); - return $interval(function() { - return JobsService.listJobs(); - }, flinkConfig["refresh-interval"]); - }); -}]).config(["$uiViewScrollProvider", function($uiViewScrollProvider) { - return $uiViewScrollProvider.useAnchorScroll(); -}]).config(["$stateProvider", "$urlRouterProvider", function($stateProvider, $urlRouterProvider) { - $stateProvider.state("overview", { - url: "/overview", - views: { - main: { - templateUrl: "partials/overview.html", - controller: 'OverviewController' - } - } - }).state("running-jobs", { - url: "/running-jobs", - views: { - main: { - templateUrl: "partials/jobs/running-jobs.html", - controller: 'RunningJobsController' - } - } - }).state("completed-jobs", { - url: "/completed-jobs", - views: { - main: { - templateUrl: "partials/jobs/completed-jobs.html", - controller: 'CompletedJobsController' - } - } - }).state("single-job", { - url: "/jobs/{jobid}", - abstract: true, - views: { - main: { - templateUrl: "partials/jobs/job.html", - controller: 'SingleJobController' - } - } - }).state("single-job.plan", { - url: "", - abstract: true, - views: { - details: { - templateUrl: "partials/jobs/job.plan.html", - controller: 'JobPlanController' - } - } - }).state("single-job.plan.overview", { - url: "", - views: { - 'node-details': { - templateUrl: "partials/jobs/job.plan.node-list.overview.html", - controller: 'JobPlanOverviewController' - } - } - }).state("single-job.plan.accumulators", { - url: "/accumulators", - views: { - 'node-details': { - templateUrl: "partials/jobs/job.plan.node-list.accumulators.html", - controller: 'JobPlanAccumulatorsController' - } - } - }).state("single-job.timeline", { - url: "/timeline", - views: { - details: { - templateUrl: "partials/jobs/job.timeline.html" - } - } - }).state("single-job.timeline.vertex", { - url: "/{vertexId}", - views: { - vertex: { - templateUrl: "partials/jobs/job.timeline.vertex.html", - controller: 'JobTimelineVertexController' - } - } - }).state("single-job.statistics", { - url: "/statistics", - views: { - details: { - templateUrl: "partials/jobs/job.statistics.html" - } - } - }).state("single-job.exceptions", { - url: "/exceptions", - views: { - details: { - templateUrl: "partials/jobs/job.exceptions.html", - controller: 'JobExceptionsController' - } - } - }).state("single-job.properties", { - url: "/properties", - views: { - details: { - templateUrl: "partials/jobs/job.properties.html", - controller: 'JobPropertiesController' - } - } - }).state("single-job.config", { - url: "/config", - views: { - details: { - templateUrl: "partials/jobs/job.config.html" - } - } - }).state("taskmanagers", { - url: "/taskmanagers", - views: { - main: { - templateUrl: "partials/taskmanagers/index.html", - controller: 'TaskManagersController' - } - } - }).state("jobmanager", { - url: "/jobmanager", - views: { - main: { - templateUrl: "partials/jobmanager/index.html" - } - } - }).state("jobmanager.config", { - url: "/config", - views: { - details: { - templateUrl: "partials/jobmanager/config.html", - controller: 'JobManagerConfigController' - } - } - }).state("jobmanager.stdout", { - url: "/stdout", - views: { - details: { - templateUrl: "partials/jobmanager/stdout.html" - } - } - }).state("jobmanager.logfile", { - url: "/logfile", - views: { - details: { - templateUrl: "partials/jobmanager/logfile.html" - } - } - }); - return $urlRouterProvider.otherwise("/overview"); -}]); - -angular.module('flinkApp').directive('bsLabel', ["JobsService", function(JobsService) { - return { - transclude: true, - replace: true, - scope: { - getLabelClass: "&", - status: "@" - }, - template: "", - link: function(scope, element, attrs) { - return scope.getLabelClass = function() { - return 'label label-' + JobsService.translateLabelState(attrs.status); - }; - } - }; -}]).directive('indicatorPrimary', ["JobsService", function(JobsService) { - return { - replace: true, - scope: { - getLabelClass: "&", - status: '@' - }, - template: "", - link: function(scope, element, attrs) { - return scope.getLabelClass = function() { - return 'fa fa-circle indicator indicator-' + JobsService.translateLabelState(attrs.status); - }; - } - }; -}]).directive('tableProperty', function() { - return { - replace: true, - scope: { - value: '=' - }, - template: "{{value || 'None'}}" - }; -}); - -angular.module('flinkApp').filter("amDurationFormatExtended", ["angularMomentConfig", function(angularMomentConfig) { - var amDurationFormatExtendedFilter; - amDurationFormatExtendedFilter = function(value, format, durationFormat) { - if (typeof value === "undefined" || value === null) { - return ""; - } - return moment.duration(value, format).format(durationFormat, { - trim: false - }); - }; - amDurationFormatExtendedFilter.$stateful = angularMomentConfig.statefulFilters; - return amDurationFormatExtendedFilter; -}]).filter("humanizeText", function() { - return function(text) { - if (text) { - return text.replace(/>/g, ">").replace(//g, ""); - } else { - return ''; - } - }; -}).filter("bytes", function() { - return function(bytes, precision) { - var number, units; - if (isNaN(parseFloat(bytes)) || !isFinite(bytes)) { - return "-"; - } - if (typeof precision === "undefined") { - precision = 1; - } - units = ["bytes", "kB", "MB", "GB", "TB", "PB"]; - number = Math.floor(Math.log(bytes) / Math.log(1024)); - return (bytes / Math.pow(1024, Math.floor(number))).toFixed(precision) + " " + units[number]; - }; -}); - -angular.module('flinkApp').service('MainService', ["$http", "flinkConfig", "$q", function($http, flinkConfig, $q) { - this.loadConfig = function() { - var deferred; - deferred = $q.defer(); - $http.get("/config").success(function(data, status, headers, config) { - return deferred.resolve(data); - }); - return deferred.promise; - }; - return this; -}]); - -angular.module('flinkApp').controller('JobManagerConfigController', ["$scope", "JobManagerConfigService", function($scope, JobManagerConfigService) { - return JobManagerConfigService.loadConfig().then(function(data) { - if ($scope.jobmanager == null) { - $scope.jobmanager = {}; - } - return $scope.jobmanager['config'] = data; - }); -}]); - -angular.module('flinkApp').service('JobManagerConfigService', ["$http", "flinkConfig", "$q", function($http, flinkConfig, $q) { - var config; - config = {}; - this.loadConfig = function() { - var deferred; - deferred = $q.defer(); - $http.get("/jobmanager/config").success(function(data, status, headers, config) { - config = data; - return deferred.resolve(data); - }); - return deferred.promise; - }; - return this; -}]); - -angular.module('flinkApp').controller('RunningJobsController', ["$scope", "$state", "$stateParams", "JobsService", function($scope, $state, $stateParams, JobsService) { - $scope.jobObserver = function() { - return $scope.jobs = JobsService.getJobs('running'); - }; - JobsService.registerObserver($scope.jobObserver); - $scope.$on('$destroy', function() { - return JobsService.unRegisterObserver($scope.jobObserver); - }); - return $scope.jobObserver(); -}]).controller('CompletedJobsController', ["$scope", "$state", "$stateParams", "JobsService", function($scope, $state, $stateParams, JobsService) { - $scope.jobObserver = function() { - return $scope.jobs = JobsService.getJobs('finished'); - }; - JobsService.registerObserver($scope.jobObserver); - $scope.$on('$destroy', function() { - return JobsService.unRegisterObserver($scope.jobObserver); - }); - return $scope.jobObserver(); -}]).controller('SingleJobController', ["$scope", "$state", "$stateParams", "JobsService", "$rootScope", "flinkConfig", "$interval", function($scope, $state, $stateParams, JobsService, $rootScope, flinkConfig, $interval) { - var refresher; - console.log('SingleJobController'); - $scope.jobid = $stateParams.jobid; - $scope.job = null; - $scope.plan = null; - $scope.vertices = null; - JobsService.loadJob($stateParams.jobid).then(function(data) { - $scope.job = data; - $scope.plan = data.plan; - return $scope.vertices = data.vertices; - }); - refresher = $interval(function() { - return JobsService.loadJob($stateParams.jobid).then(function(data) { - $scope.job = data; - return $scope.$broadcast('reload'); - }); - }, flinkConfig["refresh-interval"]); - return $scope.$on('$destroy', function() { - $scope.job = null; - $scope.plan = null; - $scope.vertices = null; - return $interval.cancel(refresher); - }); -}]).controller('JobPlanController', ["$scope", "$state", "$stateParams", "JobsService", function($scope, $state, $stateParams, JobsService) { - console.log('JobPlanController'); - $scope.nodeid = null; - $scope.nodeUnfolded = false; - $scope.stateList = JobsService.stateList(); - $scope.changeNode = function(nodeid) { - if (nodeid !== $scope.nodeid) { - $scope.nodeid = nodeid; - $scope.vertex = null; - $scope.subtasks = null; - $scope.accumulators = null; - return $scope.$broadcast('reload'); - } else { - $scope.nodeid = null; - $scope.nodeUnfolded = false; - $scope.vertex = null; - $scope.subtasks = null; - return $scope.accumulators = null; - } - }; - $scope.deactivateNode = function() { - $scope.nodeid = null; - $scope.nodeUnfolded = false; - $scope.vertex = null; - $scope.subtasks = null; - return $scope.accumulators = null; - }; - return $scope.toggleFold = function() { - return $scope.nodeUnfolded = !$scope.nodeUnfolded; - }; -}]).controller('JobPlanOverviewController', ["$scope", "JobsService", function($scope, JobsService) { - console.log('JobPlanOverviewController'); - if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.st)) { - JobsService.getSubtasks($scope.nodeid).then(function(data) { - return $scope.subtasks = data; - }); - } - return $scope.$on('reload', function(event) { - console.log('JobPlanOverviewController'); - if ($scope.nodeid) { - return JobsService.getSubtasks($scope.nodeid).then(function(data) { - return $scope.subtasks = data; - }); - } - }); -}]).controller('JobPlanAccumulatorsController', ["$scope", "JobsService", function($scope, JobsService) { - console.log('JobPlanAccumulatorsController'); - if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.accumulators)) { - JobsService.getAccumulators($scope.nodeid).then(function(data) { - $scope.accumulators = data.main; - return $scope.subtaskAccumulators = data.subtasks; - }); - } - return $scope.$on('reload', function(event) { - console.log('JobPlanAccumulatorsController'); - if ($scope.nodeid) { - return JobsService.getAccumulators($scope.nodeid).then(function(data) { - $scope.accumulators = data.main; - return $scope.subtaskAccumulators = data.subtasks; - }); - } - }); -}]).controller('JobTimelineVertexController', ["$scope", "$state", "$stateParams", "JobsService", function($scope, $state, $stateParams, JobsService) { - console.log('JobTimelineVertexController'); - JobsService.getVertex($stateParams.vertexId).then(function(data) { - return $scope.vertex = data; - }); - return $scope.$on('reload', function(event) { - console.log('JobTimelineVertexController'); - return JobsService.getVertex($stateParams.vertexId).then(function(data) { - return $scope.vertex = data; - }); - }); -}]).controller('JobExceptionsController', ["$scope", "$state", "$stateParams", "JobsService", function($scope, $state, $stateParams, JobsService) { - return JobsService.loadExceptions().then(function(data) { - return $scope.exceptions = data; - }); -}]).controller('JobPropertiesController', ["$scope", "JobsService", function($scope, JobsService) { - console.log('JobPropertiesController'); - return $scope.changeNode = function(nodeid) { - if (nodeid !== $scope.nodeid) { - $scope.nodeid = nodeid; - return JobsService.getNode(nodeid).then(function(data) { - return $scope.node = data; - }); - } else { - $scope.nodeid = null; - return $scope.node = null; - } - }; -}]); - -angular.module('flinkApp').directive('vertex', ["$state", function($state) { - return { - template: "", - scope: { - data: "=" - }, - link: function(scope, elem, attrs) { - var analyzeTime, containerW, svgEl; - svgEl = elem.children()[0]; - containerW = elem.width(); - angular.element(svgEl).attr('width', containerW); - analyzeTime = function(data) { - var chart, svg, testData; - d3.select(svgEl).selectAll("*").remove(); - testData = []; - angular.forEach(data.subtasks, function(subtask, i) { - var times; - times = [ - { - label: "Scheduled", - color: "#666", - borderColor: "#555", - starting_time: subtask.timestamps["SCHEDULED"], - ending_time: subtask.timestamps["DEPLOYING"], - type: 'regular' - }, { - label: "Deploying", - color: "#aaa", - borderColor: "#555", - starting_time: subtask.timestamps["DEPLOYING"], - ending_time: subtask.timestamps["RUNNING"], - type: 'regular' - } - ]; - if (subtask.timestamps["FINISHED"] > 0) { - times.push({ - label: "Running", - color: "#ddd", - borderColor: "#555", - starting_time: subtask.timestamps["RUNNING"], - ending_time: subtask.timestamps["FINISHED"], - type: 'regular' - }); - } - return testData.push({ - label: "(" + subtask.subtask + ") " + subtask.host, - times: times - }); - }); - chart = d3.timeline().stack().tickFormat({ - format: d3.time.format("%L"), - tickSize: 1 - }).prefix("single").labelFormat(function(label) { - return label; - }).margin({ - left: 100, - right: 0, - top: 0, - bottom: 0 - }).itemHeight(30).relativeTime(); - return svg = d3.select(svgEl).datum(testData).call(chart); - }; - analyzeTime(scope.data); - } - }; -}]).directive('timeline', ["$state", function($state) { - return { - template: "", - scope: { - vertices: "=", - jobid: "=" - }, - link: function(scope, elem, attrs) { - var analyzeTime, containerW, svgEl, translateLabel; - svgEl = elem.children()[0]; - containerW = elem.width(); - angular.element(svgEl).attr('width', containerW); - translateLabel = function(label) { - return label.replace(">", ">"); - }; - analyzeTime = function(data) { - var chart, svg, testData; - d3.select(svgEl).selectAll("*").remove(); - testData = []; - angular.forEach(data, function(vertex) { - if (vertex['start-time'] > -1) { - if (vertex.type === 'scheduled') { - return testData.push({ - times: [ - { - label: translateLabel(vertex.name), - color: "#cccccc", - borderColor: "#555555", - starting_time: vertex['start-time'], - ending_time: vertex['end-time'], - type: vertex.type - } - ] - }); - } else { - return testData.push({ - times: [ - { - label: translateLabel(vertex.name), - color: "#d9f1f7", - borderColor: "#62cdea", - starting_time: vertex['start-time'], - ending_time: vertex['end-time'], - link: vertex.id, - type: vertex.type - } - ] - }); - } - } - }); - chart = d3.timeline().stack().click(function(d, i, datum) { - if (d.link) { - return $state.go("single-job.timeline.vertex", { - jobid: scope.jobid, - vertexId: d.link - }); - } - }).tickFormat({ - format: d3.time.format("%L"), - tickSize: 1 - }).prefix("main").margin({ - left: 0, - right: 0, - top: 0, - bottom: 0 - }).itemHeight(30).showBorderLine().showHourTimeline(); - return svg = d3.select(svgEl).datum(testData).call(chart); - }; - scope.$watch(attrs.vertices, function(data) { - if (data) { - return analyzeTime(data); - } - }); - } - }; -}]).directive('jobPlan', ["$timeout", function($timeout) { - return { - template: "
", - scope: { - plan: '=', - setNode: '&' - }, - link: function(scope, elem, attrs) { - var containerW, createEdge, createLabelEdge, createLabelNode, createNode, d3mainSvg, d3mainSvgG, d3tmpSvg, drawGraph, extendLabelNodeForIteration, g, getNodeType, isSpecialIterationNode, jobid, loadJsonToDagre, mainG, mainSvgElement, mainTmpElement, mainZoom, searchForNode, shortenString, subgraphs; - g = null; - mainZoom = d3.behavior.zoom(); - subgraphs = []; - jobid = attrs.jobid; - mainSvgElement = elem.children()[0]; - mainG = elem.children().children()[0]; - mainTmpElement = elem.children()[1]; - d3mainSvg = d3.select(mainSvgElement); - d3mainSvgG = d3.select(mainG); - d3tmpSvg = d3.select(mainTmpElement); - containerW = elem.width(); - angular.element(elem.children()[0]).width(containerW); - scope.zoomIn = function() { - var translate, v1, v2; - if (mainZoom.scale() < 2.99) { - translate = mainZoom.translate(); - v1 = translate[0] * (mainZoom.scale() + 0.1 / (mainZoom.scale())); - v2 = translate[1] * (mainZoom.scale() + 0.1 / (mainZoom.scale())); - mainZoom.scale(mainZoom.scale() + 0.1); - mainZoom.translate([v1, v2]); - return d3mainSvgG.attr("transform", "translate(" + v1 + "," + v2 + ") scale(" + mainZoom.scale() + ")"); - } - }; - scope.zoomOut = function() { - var translate, v1, v2; - if (mainZoom.scale() > 0.31) { - mainZoom.scale(mainZoom.scale() - 0.1); - translate = mainZoom.translate(); - v1 = translate[0] * (mainZoom.scale() - 0.1 / (mainZoom.scale())); - v2 = translate[1] * (mainZoom.scale() - 0.1 / (mainZoom.scale())); - mainZoom.translate([v1, v2]); - return d3mainSvgG.attr("transform", "translate(" + v1 + "," + v2 + ") scale(" + mainZoom.scale() + ")"); - } - }; - createLabelEdge = function(el) { - var labelValue; - labelValue = ""; - if ((el.ship_strategy != null) || (el.local_strategy != null)) { - labelValue += "
"; - if (el.ship_strategy != null) { - labelValue += el.ship_strategy; - } - if (el.temp_mode !== undefined) { - labelValue += " (" + el.temp_mode + ")"; - } - if (el.local_strategy !== undefined) { - labelValue += ",
" + el.local_strategy; - } - labelValue += "
"; - } - return labelValue; - }; - isSpecialIterationNode = function(info) { - return info === "partialSolution" || info === "nextPartialSolution" || info === "workset" || info === "nextWorkset" || info === "solutionSet" || info === "solutionDelta"; - }; - getNodeType = function(el, info) { - if (info === "mirror") { - return 'node-mirror'; - } else if (isSpecialIterationNode(info)) { - return 'node-iteration'; - } else { - return 'node-normal'; - } - }; - createLabelNode = function(el, info, maxW, maxH) { - var labelValue, stepName; - labelValue = "
"; - if (info === "mirror") { - labelValue += "

Mirror of " + el.operator + "

"; - } else { - labelValue += "

" + el.operator + "

"; - } - if (el.description === "") { - labelValue += ""; - } else { - stepName = el.description; - stepName = shortenString(stepName); - labelValue += "

" + stepName + "

"; - } - if (el.step_function != null) { - labelValue += extendLabelNodeForIteration(el.id, maxW, maxH); - } else { - if (isSpecialIterationNode(info)) { - labelValue += "
" + info + " Node
"; - } - if (el.parallelism !== "") { - labelValue += "
Parallelism: " + el.parallelism + "
"; - } - if (el.operator !== undefined) { - labelValue += "
Operation: " + shortenString(el.operator_strategy) + "
"; - } - } - labelValue += "
"; - return labelValue; - }; - extendLabelNodeForIteration = function(id, maxW, maxH) { - var labelValue, svgID; - svgID = "svg-" + id; - labelValue = ""; - return labelValue; - }; - shortenString = function(s) { - var sbr; - if (s.charAt(0) === "<") { - s = s.replace("<", "<"); - s = s.replace(">", ">"); - } - sbr = ""; - while (s.length > 30) { - sbr = sbr + s.substring(0, 30) + "
"; - s = s.substring(30, s.length); - } - sbr = sbr + s; - return sbr; - }; - createNode = function(g, data, el, isParent, maxW, maxH) { - if (isParent == null) { - isParent = false; - } - if (el.id === data.partial_solution) { - return g.setNode(el.id, { - label: createLabelNode(el, "partialSolution", maxW, maxH), - labelType: 'html', - "class": getNodeType(el, "partialSolution") - }); - } else if (el.id === data.next_partial_solution) { - return g.setNode(el.id, { - label: createLabelNode(el, "nextPartialSolution", maxW, maxH), - labelType: 'html', - "class": getNodeType(el, "nextPartialSolution") - }); - } else if (el.id === data.workset) { - return g.setNode(el.id, { - label: createLabelNode(el, "workset", maxW, maxH), - labelType: 'html', - "class": getNodeType(el, "workset") - }); - } else if (el.id === data.next_workset) { - return g.setNode(el.id, { - label: createLabelNode(el, "nextWorkset", maxW, maxH), - labelType: 'html', - "class": getNodeType(el, "nextWorkset") - }); - } else if (el.id === data.solution_set) { - return g.setNode(el.id, { - label: createLabelNode(el, "solutionSet", maxW, maxH), - labelType: 'html', - "class": getNodeType(el, "solutionSet") - }); - } else if (el.id === data.solution_delta) { - return g.setNode(el.id, { - label: createLabelNode(el, "solutionDelta", maxW, maxH), - labelType: 'html', - "class": getNodeType(el, "solutionDelta") - }); - } else { - return g.setNode(el.id, { - label: createLabelNode(el, "", maxW, maxH), - labelType: 'html', - "class": getNodeType(el, "") - }); - } - }; - createEdge = function(g, data, el, existingNodes, pred, missingNodes) { - var missingNode; - if (existingNodes.indexOf(pred.id) !== -1) { - return g.setEdge(pred.id, el.id, { - label: createLabelEdge(pred), - labelType: 'html', - arrowhead: 'normal' - }); - } else { - missingNode = searchForNode(data, pred.id); - if (!(!missingNode || missingNodes.indexOf(missingNode.id) > -1)) { - missingNodes.push(missingNode.id); - g.setNode(missingNode.id, { - label: createLabelNode(missingNode, "mirror"), - labelType: 'html', - "class": getNodeType(missingNode, 'mirror') - }); - return g.setEdge(missingNode.id, el.id, { - label: createLabelEdge(missingNode), - labelType: 'html' - }); - } - } - }; - loadJsonToDagre = function(g, data) { - var el, existingNodes, isParent, k, l, len, len1, maxH, maxW, missingNodes, pred, r, ref, sg, toIterate; - existingNodes = []; - missingNodes = []; - if (data.nodes != null) { - toIterate = data.nodes; - } else { - toIterate = data.step_function; - isParent = true; - } - for (k = 0, len = toIterate.length; k < len; k++) { - el = toIterate[k]; - maxW = 0; - maxH = 0; - if (el.step_function) { - sg = new dagreD3.graphlib.Graph({ - multigraph: true, - compound: true - }).setGraph({ - nodesep: 20, - edgesep: 0, - ranksep: 20, - rankdir: "LR", - marginx: 10, - marginy: 10 - }); - subgraphs[el.id] = sg; - loadJsonToDagre(sg, el); - r = new dagreD3.render(); - d3tmpSvg.select('g').call(r, sg); - maxW = sg.graph().width; - maxH = sg.graph().height; - angular.element(mainTmpElement).empty(); - } - createNode(g, data, el, isParent, maxW, maxH); - existingNodes.push(el.id); - if (el.inputs != null) { - ref = el.inputs; - for (l = 0, len1 = ref.length; l < len1; l++) { - pred = ref[l]; - createEdge(g, data, el, existingNodes, pred, missingNodes); - } - } - } - return g; - }; - searchForNode = function(data, nodeID) { - var el, i, j; - for (i in data.nodes) { - el = data.nodes[i]; - if (el.id === nodeID) { - return el; - } - if (el.step_function != null) { - for (j in el.step_function) { - if (el.step_function[j].id === nodeID) { - return el.step_function[j]; - } - } - } - } - }; - drawGraph = function(data) { - var i, newScale, renderer, sg, xCenterOffset, yCenterOffset; - g = new dagreD3.graphlib.Graph({ - multigraph: true, - compound: true - }).setGraph({ - nodesep: 70, - edgesep: 0, - ranksep: 50, - rankdir: "LR", - marginx: 40, - marginy: 40 - }); - loadJsonToDagre(g, data); - renderer = new dagreD3.render(); - d3mainSvgG.call(renderer, g); - for (i in subgraphs) { - sg = subgraphs[i]; - d3mainSvg.select('svg.svg-' + i + ' g').call(renderer, sg); - } - newScale = 0.5; - xCenterOffset = Math.floor((angular.element(mainSvgElement).width() - g.graph().width * newScale) / 2); - yCenterOffset = Math.floor((angular.element(mainSvgElement).height() - g.graph().height * newScale) / 2); - mainZoom.scale(newScale).translate([xCenterOffset, yCenterOffset]); - d3mainSvgG.attr("transform", "translate(" + xCenterOffset + ", " + yCenterOffset + ") scale(" + mainZoom.scale() + ")"); - mainZoom.on("zoom", function() { - var ev; - ev = d3.event; - return d3mainSvgG.attr("transform", "translate(" + ev.translate + ") scale(" + ev.scale + ")"); - }); - mainZoom(d3mainSvg); - return d3mainSvgG.selectAll('.node').on('click', function(d) { - return scope.setNode({ - nodeid: d - }); - }); - }; - scope.$watch(attrs.plan, function(newPlan) { - if (newPlan) { - return drawGraph(newPlan); - } - }); - } - }; -}]); - -angular.module('flinkApp').service('JobsService', ["$http", "flinkConfig", "$log", "amMoment", "$q", "$timeout", function($http, flinkConfig, $log, amMoment, $q, $timeout) { - var currentJob, currentPlan, deferreds, jobObservers, jobs, notifyObservers; - currentJob = null; - currentPlan = null; - deferreds = {}; - jobs = { - running: [], - finished: [], - cancelled: [], - failed: [] - }; - jobObservers = []; - notifyObservers = function() { - return angular.forEach(jobObservers, function(callback) { - return callback(); - }); - }; - this.registerObserver = function(callback) { - return jobObservers.push(callback); - }; - this.unRegisterObserver = function(callback) { - var index; - index = jobObservers.indexOf(callback); - return jobObservers.splice(index, 1); - }; - this.stateList = function() { - return ['SCHEDULED', 'DEPLOYING', 'RUNNING', 'FINISHED', 'FAILED', 'CANCELING', 'CANCELED']; - }; - this.translateLabelState = function(state) { - switch (state.toLowerCase()) { - case 'finished': - return 'success'; - case 'failed': - return 'danger'; - case 'scheduled': - return 'default'; - case 'deploying': - return 'info'; - case 'running': - return 'primary'; - case 'canceling': - return 'warning'; - case 'pending': - return 'info'; - case 'total': - return 'black'; - default: - return 'default'; - } - }; - this.setEndTimes = function(list) { - return angular.forEach(list, function(item, jobKey) { - if (!(item['end-time'] > -1)) { - return item['end-time'] = item['start-time'] + item['duration']; - } - }); - }; - this.processVertices = function(data) { - angular.forEach(data.vertices, function(vertex, i) { - return vertex.type = 'regular'; - }); - return data.vertices.unshift({ - name: 'Scheduled', - 'start-time': data.timestamps['CREATED'], - 'end-time': data.timestamps['CREATED'] + 1, - type: 'scheduled' - }); - }; - this.listJobs = function() { - var deferred; - deferred = $q.defer(); - $http.get("/joboverview").success((function(_this) { - return function(data, status, headers, config) { - angular.forEach(data, function(list, listKey) { - switch (listKey) { - case 'running': - return jobs.running = _this.setEndTimes(list); - case 'finished': - return jobs.finished = _this.setEndTimes(list); - case 'cancelled': - return jobs.cancelled = _this.setEndTimes(list); - case 'failed': - return jobs.failed = _this.setEndTimes(list); - } - }); - deferred.resolve(jobs); - return notifyObservers(); - }; - })(this)); - return deferred.promise; - }; - this.getJobs = function(type) { - return jobs[type]; - }; - this.getAllJobs = function() { - return jobs; - }; - this.loadJob = function(jobid) { - currentJob = null; - deferreds.job = $q.defer(); - $http.get("/jobs/" + jobid).success((function(_this) { - return function(data, status, headers, config) { - _this.setEndTimes(data.vertices); - _this.processVertices(data); - return $http.get("/jobs/" + jobid + "/config").success(function(jobConfig) { - data = angular.extend(data, jobConfig); - currentJob = data; - return deferreds.job.resolve(currentJob); - }); - }; - })(this)); - return deferreds.job.promise; - }; - this.getNode = function(nodeid) { - var deferred, seekNode; - seekNode = function(nodeid, data) { - var j, len, node, sub; - for (j = 0, len = data.length; j < len; j++) { - node = data[j]; - if (node.id === nodeid) { - return node; - } - if (node.step_function) { - sub = seekNode(nodeid, node.step_function); - } - if (sub) { - return sub; - } - } - return null; - }; - deferred = $q.defer(); - deferreds.job.promise.then((function(_this) { - return function(data) { - var foundNode; - foundNode = seekNode(nodeid, currentJob.plan.nodes); - foundNode.vertex = _this.seekVertex(nodeid); - return deferred.resolve(foundNode); - }; - })(this)); - return deferred.promise; - }; - this.seekVertex = function(nodeid) { - var j, len, ref, vertex; - ref = currentJob.vertices; - for (j = 0, len = ref.length; j < len; j++) { - vertex = ref[j]; - if (vertex.id === nodeid) { - return vertex; - } - } - return null; - }; - this.getVertex = function(vertexid) { - var deferred; - deferred = $q.defer(); - deferreds.job.promise.then((function(_this) { - return function(data) { - var vertex; - vertex = _this.seekVertex(vertexid); - return $http.get("/jobs/" + currentJob.jid + "/vertices/" + vertexid + "/subtasktimes").success(function(data) { - vertex.subtasks = data.subtasks; - return deferred.resolve(vertex); - }); - }; - })(this)); - return deferred.promise; - }; - this.getSubtasks = function(vertexid) { - var deferred; - deferred = $q.defer(); - deferreds.job.promise.then((function(_this) { - return function(data) { - return $http.get("/jobs/" + currentJob.jid + "/vertices/" + vertexid).success(function(data) { - var subtasks; - subtasks = data.subtasks; - return deferred.resolve(subtasks); - }); - }; - })(this)); - return deferred.promise; - }; - this.getAccumulators = function(vertexid) { - var deferred; - deferred = $q.defer(); - deferreds.job.promise.then((function(_this) { - return function(data) { - return $http.get("/jobs/" + currentJob.jid + "/vertices/" + vertexid + "/accumulators").success(function(data) { - var accumulators; - accumulators = data['user-accumulators']; - return $http.get("/jobs/" + currentJob.jid + "/vertices/" + vertexid + "/subtasks/accumulators").success(function(data) { - var subtaskAccumulators; - subtaskAccumulators = data.subtasks; - return deferred.resolve({ - main: accumulators, - subtasks: subtaskAccumulators - }); - }); - }); - }; - })(this)); - return deferred.promise; - }; - this.loadExceptions = function() { - var deferred; - deferred = $q.defer(); - deferreds.job.promise.then((function(_this) { - return function(data) { - return $http.get("/jobs/" + currentJob.jid + "/exceptions").success(function(exceptions) { - currentJob.exceptions = exceptions; - return deferred.resolve(exceptions); - }); - }; - })(this)); - return deferred.promise; - }; - return this; -}]); - -angular.module('flinkApp').controller('OverviewController', ["$scope", "OverviewService", "JobsService", "$interval", "flinkConfig", function($scope, OverviewService, JobsService, $interval, flinkConfig) { - var refresh; - $scope.jobObserver = function() { - $scope.runningJobs = JobsService.getJobs('running'); - return $scope.finishedJobs = JobsService.getJobs('finished'); - }; - JobsService.registerObserver($scope.jobObserver); - $scope.$on('$destroy', function() { - return JobsService.unRegisterObserver($scope.jobObserver); - }); - $scope.jobObserver(); - OverviewService.loadOverview().then(function(data) { - return $scope.overview = data; - }); - refresh = $interval(function() { - return OverviewService.loadOverview().then(function(data) { - return $scope.overview = data; - }); - }, flinkConfig["refresh-interval"]); - return $scope.$on('$destroy', function() { - return $interval.cancel(refresh); - }); -}]); - -angular.module('flinkApp').service('OverviewService', ["$http", "flinkConfig", "$q", function($http, flinkConfig, $q) { - var overview; - overview = {}; - this.loadOverview = function() { - var deferred; - deferred = $q.defer(); - $http.get("/overview").success(function(data, status, headers, config) { - overview = data; - return deferred.resolve(data); - }); - return deferred.promise; - }; - return this; -}]); - -angular.module('flinkApp').controller('TaskManagersController', ["$scope", "TaskManagersService", "$interval", "flinkConfig", function($scope, TaskManagersService, $interval, flinkConfig) { - var refresh; - TaskManagersService.loadManagers().then(function(data) { - return $scope.managers = data; - }); - refresh = $interval(function() { - return TaskManagersService.loadManagers().then(function(data) { - return $scope.managers = data; - }); - }, flinkConfig["refresh-interval"]); - $scope.$on('$destroy', function() { - return $interval.cancel(refresh); - }); - $scope.managerId = null; - return $scope.changeManager = function(managerId) { - if (managerId !== $scope.managerId) { - return $scope.managerId = managerId; - } else { - return $scope.managerId = null; - } - }; -}]); - -angular.module('flinkApp').service('TaskManagersService', ["$http", "flinkConfig", "$q", function($http, flinkConfig, $q) { - this.loadManagers = function() { - var deferred; - deferred = $q.defer(); - $http.get("/taskmanagers").success(function(data, status, headers, config) { - return deferred.resolve(data['taskmanagers']); - }); - return deferred.promise; - }; - return this; -}]); - -<<<<<<< HEAD:flink-runtime-web/web-dashboard/web/js/index.js -//# 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/overview/overview.ctrl.coffee","modules/overview/overview.ctrl.js","modules/overview/overview.svc.coffee","modules/overview/overview.svc.js","modules/taskmanagers/taskmanagers.ctrl.coffee","modules/taskmanagers/taskmanagers.ctrl.js","modules/taskmanagers/taskmanagers.svc.coffee","modules/taskmanagers/taskmanagers.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'])\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  \"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.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    abstract: true\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.plan.html\"\n        controller: 'JobPlanController'\n\n  .state \"single-job.plan.overview\",\n    url: \"\"\n    views:\n      'node-details':\n        templateUrl: \"partials/jobs/job.plan.node-list.overview.html\"\n        controller: 'JobPlanOverviewController' \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.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.statistics\",\n    url: \"/statistics\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.statistics.html\"\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.properties\",\n    url: \"/properties\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.properties.html\"\n        controller: 'JobPropertiesController'\n\n  .state \"single-job.config\",\n    url: \"/config\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.config.html\"\n\n  .state \"taskmanagers\",\n    url: \"/taskmanagers\"\n    views:\n      main:\n        templateUrl: \"partials/taskmanagers/index.html\"\n        controller: 'TaskManagersController'\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\n  .state \"jobmanager.logfile\",\n    url: \"/logfile\"\n    views:\n      details:\n        templateUrl: \"partials/jobmanager/logfile.html\"\n\n  $urlRouterProvider.otherwise \"/overview\"\n","angular.module('flinkApp', ['ui.router', 'angularMoment']).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  \"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}).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    abstract: true,\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.plan.html\",\n        controller: 'JobPlanController'\n      }\n    }\n  }).state(\"single-job.plan.overview\", {\n    url: \"\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.overview.html\",\n        controller: 'JobPlanOverviewController'\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.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.statistics\", {\n    url: \"/statistics\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.statistics.html\"\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.properties\", {\n    url: \"/properties\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.properties.html\",\n        controller: 'JobPropertiesController'\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(\"taskmanagers\", {\n    url: \"/taskmanagers\",\n    views: {\n      main: {\n        templateUrl: \"partials/taskmanagers/index.html\",\n        controller: 'TaskManagersController'\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      }\n    }\n  }).state(\"jobmanager.logfile\", {\n    url: \"/logfile\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobmanager/logfile.html\"\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 '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('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 \"humanizeText\", ->\n  (text) ->\n    # TODO: extend... a lot\n    if text then text.replace(/&gt;/g, \">\").replace(/<br\\/>/g,\"\") else ''\n\n.filter \"bytes\", ->\n  (bytes, precision) ->\n    return \"-\"  if isNaN(parseFloat(bytes)) or not isFinite(bytes)\n    precision = 1  if typeof precision is \"undefined\"\n    units = [ \"bytes\", \"kB\", \"MB\", \"GB\", \"TB\", \"PB\" ]\n    number = Math.floor(Math.log(bytes) / Math.log(1024))\n    (bytes / Math.pow(1024, Math.floor(number))).toFixed(precision) + \" \" + units[number]\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(\"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(\"bytes\", function() {\n  return function(bytes, precision) {\n    var number, units;\n    if (isNaN(parseFloat(bytes)) || !isFinite(bytes)) {\n      return \"-\";\n    }\n    if (typeof precision === \"undefined\") {\n      precision = 1;\n    }\n    units = [\"bytes\", \"kB\", \"MB\", \"GB\", \"TB\", \"PB\"];\n    number = Math.floor(Math.log(bytes) / Math.log(1024));\n    return (bytes / Math.pow(1024, Math.floor(number))).toFixed(precision) + \" \" + units[number];\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 \"/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(\"/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","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});\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(\"/jobmanager/config\")\n    .success (data, status, headers, config) ->\n      config = 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(\"/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});\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, $rootScope, flinkConfig, $interval) ->\n  console.log 'SingleJobController'\n\n  $scope.jobid = $stateParams.jobid\n  $scope.job = null\n  $scope.plan = null\n  $scope.vertices = null\n\n  JobsService.loadJob($stateParams.jobid).then (data) ->\n    $scope.job = data\n    $scope.plan = data.plan\n    $scope.vertices = 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\n    $interval.cancel(refresher)\n\n\n# --------------------------------------\n\n.controller 'JobPlanController', ($scope, $state, $stateParams, JobsService) ->\n  console.log 'JobPlanController'\n\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\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.deactivateNode = ->\n    $scope.nodeid = null\n    $scope.nodeUnfolded = false\n    $scope.vertex = null\n    $scope.subtasks = null\n    $scope.accumulators = null\n\n  $scope.toggleFold = ->\n    $scope.nodeUnfolded = !$scope.nodeUnfolded\n\n# --------------------------------------\n\n.controller 'JobPlanOverviewController', ($scope, JobsService) ->\n  console.log 'JobPlanOverviewController'\n\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.st)\n    JobsService.getSubtasks($scope.nodeid).then (data) ->\n      $scope.subtasks = data\n\n  $scope.$on 'reload', (event) ->\n    console.log 'JobPlanOverviewController'\n    if $scope.nodeid\n      JobsService.getSubtasks($scope.nodeid).then (data) ->\n        $scope.subtasks = data\n\n# --------------------------------------\n\n.controller 'JobPlanAccumulatorsController', ($scope, JobsService) ->\n  console.log 'JobPlanAccumulatorsController'\n\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.accumulators)\n    JobsService.getAccumulators($scope.nodeid).then (data) ->\n      $scope.accumulators = data.main\n      $scope.subtaskAccumulators = data.subtasks\n\n  $scope.$on 'reload', (event) ->\n    console.log 'JobPlanAccumulatorsController'\n    if $scope.nodeid\n      JobsService.getAccumulators($scope.nodeid).then (data) ->\n        $scope.accumulators = data.main\n        $scope.subtaskAccumulators = data.subtasks\n\n# --------------------------------------\n\n.controller 'JobTimelineVertexController', ($scope, $state, $stateParams, JobsService) ->\n  console.log 'JobTimelineVertexController'\n\n  JobsService.getVertex($stateParams.vertexId).then (data) ->\n    $scope.vertex = data\n\n  $scope.$on 'reload', (event) ->\n    console.log 'JobTimelineVertexController'\n    JobsService.getVertex($stateParams.vertexId).then (data) ->\n      $scope.vertex = data\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  console.log 'JobPropertiesController'\n\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","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, $rootScope, flinkConfig, $interval) {\n  var refresher;\n  console.log('SingleJobController');\n  $scope.jobid = $stateParams.jobid;\n  $scope.job = null;\n  $scope.plan = null;\n  $scope.vertices = null;\n  JobsService.loadJob($stateParams.jobid).then(function(data) {\n    $scope.job = data;\n    $scope.plan = data.plan;\n    return $scope.vertices = 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  return $scope.$on('$destroy', function() {\n    $scope.job = null;\n    $scope.plan = null;\n    $scope.vertices = null;\n    return $interval.cancel(refresher);\n  });\n}).controller('JobPlanController', function($scope, $state, $stateParams, JobsService) {\n  console.log('JobPlanController');\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      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.deactivateNode = function() {\n    $scope.nodeid = null;\n    $scope.nodeUnfolded = false;\n    $scope.vertex = null;\n    $scope.subtasks = null;\n    return $scope.accumulators = null;\n  };\n  return $scope.toggleFold = function() {\n    return $scope.nodeUnfolded = !$scope.nodeUnfolded;\n  };\n}).controller('JobPlanOverviewController', function($scope, JobsService) {\n  console.log('JobPlanOverviewController');\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.st)) {\n    JobsService.getSubtasks($scope.nodeid).then(function(data) {\n      return $scope.subtasks = data;\n    });\n  }\n  return $scope.$on('reload', function(event) {\n    console.log('JobPlanOverviewController');\n    if ($scope.nodeid) {\n      return JobsService.getSubtasks($scope.nodeid).then(function(data) {\n        return $scope.subtasks = data;\n      });\n    }\n  });\n}).controller('JobPlanAccumulatorsController', function($scope, JobsService) {\n  console.log('JobPlanAccumulatorsController');\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.accumulators)) {\n    JobsService.getAccumulators($scope.nodeid).then(function(data) {\n      $scope.accumulators = data.main;\n      return $scope.subtaskAccumulators = data.subtasks;\n    });\n  }\n  return $scope.$on('reload', function(event) {\n    console.log('JobPlanAccumulatorsController');\n    if ($scope.nodeid) {\n      return JobsService.getAccumulators($scope.nodeid).then(function(data) {\n        $scope.accumulators = data.main;\n        return $scope.subtaskAccumulators = data.subtasks;\n      });\n    }\n  });\n}).controller('JobTimelineVertexController', function($scope, $state, $stateParams, JobsService) {\n  console.log('JobTimelineVertexController');\n  JobsService.getVertex($stateParams.vertexId).then(function(data) {\n    return $scope.vertex = data;\n  });\n  return $scope.$on('reload', function(event) {\n    console.log('JobTimelineVertexController');\n    return JobsService.getVertex($stateParams.vertexId).then(function(data) {\n      return $scope.vertex = data;\n    });\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  console.log('JobPropertiesController');\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});\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\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`\n      \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, missingNodes) ->\n      unless existingNodes.indexOf(pred.id) is -1\n        g.setEdge pred.id, el.id,\n          label: createLabelEdge(pred)\n          labelType: 'html'\n          arrowhead: 'normal'\n\n      else\n        missingNode = searchForNode(data, pred.id)\n\n        unless !missingNode or missingNodes.indexOf(missingNode.id) > -1\n          missingNodes.push(missingNode.id)\n          g.setNode missingNode.id,\n            label: createLabelNode(missingNode, \"mirror\")\n            labelType: 'html'\n            class: getNodeType(missingNode, 'mirror')\n\n          g.setEdge missingNode.id, el.id,\n            label: createLabelEdge(missingNode)\n            labelType: 'html'\n\n    loadJsonToDagre = (g, data) ->\n      existingNodes = []\n      missingNodes = []\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, missingNodes)\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('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) {\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, missingNodes) {\n        var missingNode;\n        if (existingNodes.indexOf(pred.id) !== -1) {\n          return g.setEdge(pred.id, el.id, {\n            label: createLabelEdge(pred),\n            labelType: 'html',\n            arrowhead: 'normal'\n          });\n        } else {\n          missingNode = searchForNode(data, pred.id);\n          if (!(!missingNode || missingNodes.indexOf(missingNode.id) > -1)) {\n            missingNodes.push(missingNode.id);\n            g.setNode(missingNode.id, {\n              label: createLabelNode(missingNode, \"mirror\"),\n              labelType: 'html',\n              \"class\": getNodeType(missingNode, 'mirror')\n            });\n            return g.setEdge(missingNode.id, el.id, {\n              label: createLabelEdge(missingNode),\n              labelType: 'html'\n            });\n          }\n        }\n      };\n      loadJsonToDagre = function(g, data) {\n        var el, existingNodes, isParent, k, l, len, len1, maxH, maxW, missingNodes, pred, r, ref, sg, toIterate;\n        existingNodes = [];\n        missingNodes = [];\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, missingNodes);\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 \"/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 \"/jobs/\" + jobid\n    .success (data, status, headers, config) =>\n      @setEndTimes(data.vertices)\n      @processVertices(data)\n\n      $http.get \"/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 \"/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 \"/jobs/\" + currentJob.jid + \"/vertices/\" + vertexid\n      .success (data) ->\n        subtasks = data.subtasks\n\n        deferred.resolve(subtasks)\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 \"/jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/accumulators\"\n      .success (data) ->\n        accumulators = data['user-accumulators']\n\n        $http.get \"/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  @loadExceptions = ->\n    deferred = $q.defer()\n\n    deferreds.job.promise.then (data) =>\n\n      $http.get \"/jobs/\" + currentJob.jid + \"/exceptions\"\n      .success (exceptions) ->\n        currentJob.exceptions = exceptions\n\n        deferred.resolve(exceptions)\n\n    deferred.promise\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(\"/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(\"/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(\"/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(\"/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(\"/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.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(\"/jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/accumulators\").success(function(data) {\n          var accumulators;\n          accumulators = data['user-accumulators'];\n          return $http.get(\"/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.loadExceptions = function() {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(\"/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  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(\"/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(\"/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 'TaskManagersController', ($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  $scope.managerId = null\n\n  $scope.changeManager = (managerId) ->\n    if managerId != $scope.managerId\n      $scope.managerId = managerId\n    else\n      $scope.managerId = null\n","angular.module('flinkApp').controller('TaskManagersController', 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  $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n  $scope.managerId = null;\n  return $scope.changeManager = function(managerId) {\n    if (managerId !== $scope.managerId) {\n      return $scope.managerId = managerId;\n    } else {\n      return $scope.managerId = null;\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 'TaskManagersService', ($http, flinkConfig, $q) ->\n  @loadManagers = ->\n    deferred = $q.defer()\n\n    $http.get(\"/taskmanagers\")\n    .success (data, status, headers, config) ->\n      deferred.resolve(data['taskmanagers'])\n\n    deferred.promise\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(\"/taskmanagers\").success(function(data, status, headers, config) {\n      return deferred.resolve(data['taskmanagers']);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n"],"sourceRoot":"/source/"} -======= -//# 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/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/overview/overview.ctrl.coffee","modules/overview/overview.ctrl.js","modules/overview/overview.svc.coffee","modules/overview/overview.svc.js","modules/taskmanagers/taskmanagers.ctrl.coffee","modules/taskmanagers/taskmanagers.ctrl.js","modules/taskmanagers/taskmanagers.svc.coffee","modules/taskmanagers/taskmanagers.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'])\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  \"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.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    abstract: true\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.plan.html\"\n        controller: 'JobPlanController'\n\n  .state \"single-job.plan.overview\",\n    url: \"\"\n    views:\n      'node-details':\n        templateUrl: \"partials/jobs/job.plan.node-list.overview.html\"\n        controller: 'JobPlanOverviewController' \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.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.statistics\",\n    url: \"/statistics\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.statistics.html\"\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.properties\",\n    url: \"/properties\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.properties.html\"\n        controller: 'JobPropertiesController'\n\n  .state \"single-job.config\",\n    url: \"/config\"\n    views:\n      details:\n        templateUrl: \"partials/jobs/job.config.html\"\n\n  .state \"taskmanagers\",\n    url: \"/taskmanagers\"\n    views:\n      main:\n        templateUrl: \"partials/taskmanagers/index.html\"\n        controller: 'TaskManagersController'\n\n  $urlRouterProvider.otherwise \"/overview\"\n","angular.module('flinkApp', ['ui.router', 'angularMoment']).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  \"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}).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    abstract: true,\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.plan.html\",\n        controller: 'JobPlanController'\n      }\n    }\n  }).state(\"single-job.plan.overview\", {\n    url: \"\",\n    views: {\n      'node-details': {\n        templateUrl: \"partials/jobs/job.plan.node-list.overview.html\",\n        controller: 'JobPlanOverviewController'\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.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.statistics\", {\n    url: \"/statistics\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.statistics.html\"\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.properties\", {\n    url: \"/properties\",\n    views: {\n      details: {\n        templateUrl: \"partials/jobs/job.properties.html\",\n        controller: 'JobPropertiesController'\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(\"taskmanagers\", {\n    url: \"/taskmanagers\",\n    views: {\n      main: {\n        templateUrl: \"partials/taskmanagers/index.html\",\n        controller: 'TaskManagersController'\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 '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('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 \"humanizeText\", ->\n  (text) ->\n    # TODO: extend... a lot\n    if text then text.replace(/&gt;/g, \">\").replace(/<br\\/>/g,\"\") else ''\n\n.filter \"bytes\", ->\n  (bytes, precision) ->\n    return \"-\"  if isNaN(parseFloat(bytes)) or not isFinite(bytes)\n    precision = 1  if typeof precision is \"undefined\"\n    units = [ \"bytes\", \"kB\", \"MB\", \"GB\", \"TB\", \"PB\" ]\n    number = Math.floor(Math.log(bytes) / Math.log(1024))\n    (bytes / Math.pow(1024, Math.floor(number))).toFixed(precision) + \" \" + units[number]\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(\"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(\"bytes\", function() {\n  return function(bytes, precision) {\n    var number, units;\n    if (isNaN(parseFloat(bytes)) || !isFinite(bytes)) {\n      return \"-\";\n    }\n    if (typeof precision === \"undefined\") {\n      precision = 1;\n    }\n    units = [\"bytes\", \"kB\", \"MB\", \"GB\", \"TB\", \"PB\"];\n    number = Math.floor(Math.log(bytes) / Math.log(1024));\n    return (bytes / Math.pow(1024, Math.floor(number))).toFixed(precision) + \" \" + units[number];\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 \"/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(\"/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 '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, $rootScope, flinkConfig, $interval) ->\n  console.log 'SingleJobController'\n\n  $scope.jobid = $stateParams.jobid\n  $scope.job = null\n  $scope.plan = null\n  $scope.vertices = null\n\n  JobsService.loadJob($stateParams.jobid).then (data) ->\n    $scope.job = data\n    $scope.plan = data.plan\n    $scope.vertices = 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\n    $interval.cancel(refresher)\n\n\n# --------------------------------------\n\n.controller 'JobPlanController', ($scope, $state, $stateParams, JobsService) ->\n  console.log 'JobPlanController'\n\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\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.deactivateNode = ->\n    $scope.nodeid = null\n    $scope.nodeUnfolded = false\n    $scope.vertex = null\n    $scope.subtasks = null\n    $scope.accumulators = null\n\n  $scope.toggleFold = ->\n    $scope.nodeUnfolded = !$scope.nodeUnfolded\n\n# --------------------------------------\n\n.controller 'JobPlanOverviewController', ($scope, JobsService) ->\n  console.log 'JobPlanOverviewController'\n\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.st)\n    JobsService.getSubtasks($scope.nodeid).then (data) ->\n      $scope.subtasks = data\n\n  $scope.$on 'reload', (event) ->\n    console.log 'JobPlanOverviewController'\n    if $scope.nodeid\n      JobsService.getSubtasks($scope.nodeid).then (data) ->\n        $scope.subtasks = data\n\n# --------------------------------------\n\n.controller 'JobPlanAccumulatorsController', ($scope, JobsService) ->\n  console.log 'JobPlanAccumulatorsController'\n\n  if $scope.nodeid and (!$scope.vertex or !$scope.vertex.accumulators)\n    JobsService.getAccumulators($scope.nodeid).then (data) ->\n      $scope.accumulators = data.main\n      $scope.subtaskAccumulators = data.subtasks\n\n  $scope.$on 'reload', (event) ->\n    console.log 'JobPlanAccumulatorsController'\n    if $scope.nodeid\n      JobsService.getAccumulators($scope.nodeid).then (data) ->\n        $scope.accumulators = data.main\n        $scope.subtaskAccumulators = data.subtasks\n\n# --------------------------------------\n\n.controller 'JobTimelineVertexController', ($scope, $state, $stateParams, JobsService) ->\n  console.log 'JobTimelineVertexController'\n\n  JobsService.getVertex($stateParams.vertexId).then (data) ->\n    $scope.vertex = data\n\n  $scope.$on 'reload', (event) ->\n    console.log 'JobTimelineVertexController'\n    JobsService.getVertex($stateParams.vertexId).then (data) ->\n      $scope.vertex = data\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  console.log 'JobPropertiesController'\n\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","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, $rootScope, flinkConfig, $interval) {\n  var refresher;\n  console.log('SingleJobController');\n  $scope.jobid = $stateParams.jobid;\n  $scope.job = null;\n  $scope.plan = null;\n  $scope.vertices = null;\n  JobsService.loadJob($stateParams.jobid).then(function(data) {\n    $scope.job = data;\n    $scope.plan = data.plan;\n    return $scope.vertices = 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  return $scope.$on('$destroy', function() {\n    $scope.job = null;\n    $scope.plan = null;\n    $scope.vertices = null;\n    return $interval.cancel(refresher);\n  });\n}).controller('JobPlanController', function($scope, $state, $stateParams, JobsService) {\n  console.log('JobPlanController');\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      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.deactivateNode = function() {\n    $scope.nodeid = null;\n    $scope.nodeUnfolded = false;\n    $scope.vertex = null;\n    $scope.subtasks = null;\n    return $scope.accumulators = null;\n  };\n  return $scope.toggleFold = function() {\n    return $scope.nodeUnfolded = !$scope.nodeUnfolded;\n  };\n}).controller('JobPlanOverviewController', function($scope, JobsService) {\n  console.log('JobPlanOverviewController');\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.st)) {\n    JobsService.getSubtasks($scope.nodeid).then(function(data) {\n      return $scope.subtasks = data;\n    });\n  }\n  return $scope.$on('reload', function(event) {\n    console.log('JobPlanOverviewController');\n    if ($scope.nodeid) {\n      return JobsService.getSubtasks($scope.nodeid).then(function(data) {\n        return $scope.subtasks = data;\n      });\n    }\n  });\n}).controller('JobPlanAccumulatorsController', function($scope, JobsService) {\n  console.log('JobPlanAccumulatorsController');\n  if ($scope.nodeid && (!$scope.vertex || !$scope.vertex.accumulators)) {\n    JobsService.getAccumulators($scope.nodeid).then(function(data) {\n      $scope.accumulators = data.main;\n      return $scope.subtaskAccumulators = data.subtasks;\n    });\n  }\n  return $scope.$on('reload', function(event) {\n    console.log('JobPlanAccumulatorsController');\n    if ($scope.nodeid) {\n      return JobsService.getAccumulators($scope.nodeid).then(function(data) {\n        $scope.accumulators = data.main;\n        return $scope.subtaskAccumulators = data.subtasks;\n      });\n    }\n  });\n}).controller('JobTimelineVertexController', function($scope, $state, $stateParams, JobsService) {\n  console.log('JobTimelineVertexController');\n  JobsService.getVertex($stateParams.vertexId).then(function(data) {\n    return $scope.vertex = data;\n  });\n  return $scope.$on('reload', function(event) {\n    console.log('JobTimelineVertexController');\n    return JobsService.getVertex($stateParams.vertexId).then(function(data) {\n      return $scope.vertex = data;\n    });\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  console.log('JobPropertiesController');\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});\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\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`\n      \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, missingNodes) ->\n      unless existingNodes.indexOf(pred.id) is -1\n        g.setEdge pred.id, el.id,\n          label: createLabelEdge(pred)\n          labelType: 'html'\n          arrowhead: 'normal'\n\n      else\n        missingNode = searchForNode(data, pred.id)\n\n        unless !missingNode or missingNodes.indexOf(missingNode.id) > -1\n          missingNodes.push(missingNode.id)\n          g.setNode missingNode.id,\n            label: createLabelNode(missingNode, \"mirror\")\n            labelType: 'html'\n            class: getNodeType(missingNode, 'mirror')\n\n          g.setEdge missingNode.id, el.id,\n            label: createLabelEdge(missingNode)\n            labelType: 'html'\n\n    loadJsonToDagre = (g, data) ->\n      existingNodes = []\n      missingNodes = []\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, missingNodes)\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('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) {\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, missingNodes) {\n        var missingNode;\n        if (existingNodes.indexOf(pred.id) !== -1) {\n          return g.setEdge(pred.id, el.id, {\n            label: createLabelEdge(pred),\n            labelType: 'html',\n            arrowhead: 'normal'\n          });\n        } else {\n          missingNode = searchForNode(data, pred.id);\n          if (!(!missingNode || missingNodes.indexOf(missingNode.id) > -1)) {\n            missingNodes.push(missingNode.id);\n            g.setNode(missingNode.id, {\n              label: createLabelNode(missingNode, \"mirror\"),\n              labelType: 'html',\n              \"class\": getNodeType(missingNode, 'mirror')\n            });\n            return g.setEdge(missingNode.id, el.id, {\n              label: createLabelEdge(missingNode),\n              labelType: 'html'\n            });\n          }\n        }\n      };\n      loadJsonToDagre = function(g, data) {\n        var el, existingNodes, isParent, k, l, len, len1, maxH, maxW, missingNodes, pred, r, ref, sg, toIterate;\n        existingNodes = [];\n        missingNodes = [];\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, missingNodes);\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 \"/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 \"/jobs/\" + jobid\n    .success (data, status, headers, config) =>\n      @setEndTimes(data.vertices)\n      @processVertices(data)\n\n      $http.get \"/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 \"/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 \"/jobs/\" + currentJob.jid + \"/vertices/\" + vertexid\n      .success (data) ->\n        subtasks = data.subtasks\n\n        deferred.resolve(subtasks)\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 \"/jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/accumulators\"\n      .success (data) ->\n        accumulators = data['user-accumulators']\n\n        $http.get \"/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  @loadExceptions = ->\n    deferred = $q.defer()\n\n    deferreds.job.promise.then (data) =>\n\n      $http.get \"/jobs/\" + currentJob.jid + \"/exceptions\"\n      .success (exceptions) ->\n        currentJob.exceptions = exceptions\n\n        deferred.resolve(exceptions)\n\n    deferred.promise\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(\"/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(\"/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(\"/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(\"/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(\"/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.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(\"/jobs/\" + currentJob.jid + \"/vertices/\" + vertexid + \"/accumulators\").success(function(data) {\n          var accumulators;\n          accumulators = data['user-accumulators'];\n          return $http.get(\"/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.loadExceptions = function() {\n    var deferred;\n    deferred = $q.defer();\n    deferreds.job.promise.then((function(_this) {\n      return function(data) {\n        return $http.get(\"/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  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(\"/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(\"/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 'TaskManagersController', ($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  $scope.managerId = null\n\n  $scope.changeManager = (managerId) ->\n    if managerId != $scope.managerId\n      $scope.managerId = managerId\n    else\n      $scope.managerId = null\n","angular.module('flinkApp').controller('TaskManagersController', 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  $scope.$on('$destroy', function() {\n    return $interval.cancel(refresh);\n  });\n  $scope.managerId = null;\n  return $scope.changeManager = function(managerId) {\n    if (managerId !== $scope.managerId) {\n      return $scope.managerId = managerId;\n    } else {\n      return $scope.managerId = null;\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 'TaskManagersService', ($http, flinkConfig, $q) ->\n  @loadManagers = ->\n    deferred = $q.defer()\n\n    $http.get(\"/taskmanagers\")\n    .success (data, status, headers, config) ->\n      deferred.resolve(data['taskmanagers'])\n\n    deferred.promise\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(\"/taskmanagers\").success(function(data, status, headers, config) {\n      return deferred.resolve(data['taskmanagers']);\n    });\n    return deferred.promise;\n  };\n  return this;\n});\n"],"sourceRoot":"/source/"} ->>>>>>> [FLINK-2793] [runtime-web] Redirect to leader in non-standalone mode:flink-runtime-web/src/main/resources/web/js/index.js diff --git a/flink-runtime-web/src/test/java/org/apache/flink/runtime/webmonitor/WebRuntimeMonitorITCase.java b/flink-runtime-web/src/test/java/org/apache/flink/runtime/webmonitor/WebRuntimeMonitorITCase.java index 5167d1393cee2314f14793828994e75086c85b00..68b00dc682b9ddbd53724b4039918bb7c2fd6759 100644 --- a/flink-runtime-web/src/test/java/org/apache/flink/runtime/webmonitor/WebRuntimeMonitorITCase.java +++ b/flink-runtime-web/src/test/java/org/apache/flink/runtime/webmonitor/WebRuntimeMonitorITCase.java @@ -46,6 +46,7 @@ import scala.concurrent.duration.Deadline; import scala.concurrent.duration.FiniteDuration; import java.io.File; +import java.nio.file.Files; import java.util.ArrayList; import java.util.List; import java.util.Scanner; @@ -62,7 +63,7 @@ public class WebRuntimeMonitorITCase extends TestLogger { private final static FiniteDuration TestTimeout = new FiniteDuration(2, TimeUnit.MINUTES); - private final String MAIN_RESOURCES_PATH = getClass().getResource("/../classes/web").getPath(); + private final String MAIN_RESOURCES_PATH = getClass().getResource("/web").getPath(); /** * Tests operation of the monitor in standalone operation. @@ -82,10 +83,13 @@ public class WebRuntimeMonitorITCase extends TestLogger { ActorSystem jmActorSystem = flink.jobManagerActorSystems().get().head(); ActorRef jmActor = flink.jobManagerActors().get().head(); + File logDir = temporaryFolder.newFolder("log"); + Files.createFile(new File(logDir, "jobmanager.log").toPath()); + Files.createFile(new File(logDir, "jobmanager.out").toPath()); + Configuration monitorConfig = new Configuration(); - monitorConfig.setString(WebMonitorConfig.JOB_MANAGER_WEB_DOC_ROOT_KEY, MAIN_RESOURCES_PATH); - monitorConfig.setBoolean(ConfigConstants.JOB_MANAGER_NEW_WEB_FRONTEND_KEY, true); monitorConfig.setInteger(ConfigConstants.JOB_MANAGER_WEB_PORT_KEY, 0); + monitorConfig.setString(ConfigConstants.JOB_MANAGER_WEB_LOG_PATH_KEY, logDir.getAbsolutePath()); // Needs to match the leader address from the leader retrieval service String jobManagerAddress = AkkaUtils.getAkkaURL(jmActorSystem, jmActor); @@ -143,9 +147,13 @@ public class WebRuntimeMonitorITCase extends TestLogger { final Configuration config = ZooKeeperTestUtils.createZooKeeperRecoveryModeConfig( zooKeeper.getConnectString(), temporaryFolder.getRoot().getPath()); - config.setString(WebMonitorConfig.JOB_MANAGER_WEB_DOC_ROOT_KEY, MAIN_RESOURCES_PATH); - config.setBoolean(ConfigConstants.JOB_MANAGER_NEW_WEB_FRONTEND_KEY, true); + + File logDir = temporaryFolder.newFolder(); + Files.createFile(new File(logDir, "jobmanager.log").toPath()); + Files.createFile(new File(logDir, "jobmanager.out").toPath()); + config.setInteger(ConfigConstants.JOB_MANAGER_WEB_PORT_KEY, 0); + config.setString(ConfigConstants.JOB_MANAGER_WEB_LOG_PATH_KEY, logDir.getAbsolutePath()); for (int i = 0; i < jobManagerSystem.length; i++) { jobManagerSystem[i] = AkkaUtils.createActorSystem(new Configuration(), @@ -280,10 +288,13 @@ public class WebRuntimeMonitorITCase extends TestLogger { try (TestingServer zooKeeper = new TestingServer()) { + File logDir = temporaryFolder.newFolder(); + Files.createFile(new File(logDir, "jobmanager.log").toPath()); + Files.createFile(new File(logDir, "jobmanager.out").toPath()); + final Configuration config = new Configuration(); - config.setString(WebMonitorConfig.JOB_MANAGER_WEB_DOC_ROOT_KEY, MAIN_RESOURCES_PATH); - config.setBoolean(ConfigConstants.JOB_MANAGER_NEW_WEB_FRONTEND_KEY, true); config.setInteger(ConfigConstants.JOB_MANAGER_WEB_PORT_KEY, 0); + config.setString(ConfigConstants.JOB_MANAGER_WEB_LOG_PATH_KEY, logDir.getAbsolutePath()); config.setString(ConfigConstants.RECOVERY_MODE, "ZOOKEEPER"); config.setString(ConfigConstants.ZOOKEEPER_QUORUM_KEY, zooKeeper.getConnectString()); diff --git a/flink-runtime-web/web-dashboard/gulpfile.js b/flink-runtime-web/web-dashboard/gulpfile.js index 2abfccecf02290ee1f39e9c9407fd47172acbe6a..737519ef92237ec30b8c6323c0aeef62199a9ea0 100644 --- a/flink-runtime-web/web-dashboard/gulpfile.js +++ b/flink-runtime-web/web-dashboard/gulpfile.js @@ -43,7 +43,7 @@ var path = require('path'); var environment = 'development'; var paths = { src: './app/', - dest: '../src/main/resources/web/', + dest: './web/', vendor: './bower_components/', vendorLocal: './vendor-local/', assets: './assets/', @@ -168,7 +168,7 @@ gulp.task('watch', function () { }); gulp.task('serve', serve({ - root: '../src/main/resources/web/', + root: 'web', port: 3001 })); diff --git a/flink-runtime-web/src/main/resources/web/css/index.css b/flink-runtime-web/web-dashboard/web/css/index.css similarity index 100% rename from flink-runtime-web/src/main/resources/web/css/index.css rename to flink-runtime-web/web-dashboard/web/css/index.css diff --git a/flink-runtime-web/src/main/resources/web/css/vendor.css b/flink-runtime-web/web-dashboard/web/css/vendor.css similarity index 100% rename from flink-runtime-web/src/main/resources/web/css/vendor.css rename to flink-runtime-web/web-dashboard/web/css/vendor.css diff --git a/flink-runtime-web/src/main/resources/web/fonts/FontAwesome.otf b/flink-runtime-web/web-dashboard/web/fonts/FontAwesome.otf similarity index 100% rename from flink-runtime-web/src/main/resources/web/fonts/FontAwesome.otf rename to flink-runtime-web/web-dashboard/web/fonts/FontAwesome.otf diff --git a/flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.eot b/flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.eot similarity index 100% rename from flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.eot rename to flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.eot diff --git a/flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.svg b/flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.svg similarity index 100% rename from flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.svg rename to flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.svg diff --git a/flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.ttf b/flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.ttf similarity index 100% rename from flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.ttf rename to flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.ttf diff --git a/flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.woff b/flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.woff similarity index 100% rename from flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.woff rename to flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.woff diff --git a/flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.woff2 b/flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.woff2 similarity index 100% rename from flink-runtime-web/src/main/resources/web/fonts/fontawesome-webfont.woff2 rename to flink-runtime-web/web-dashboard/web/fonts/fontawesome-webfont.woff2 diff --git a/flink-runtime-web/src/main/resources/web/images/flink-logo.png b/flink-runtime-web/web-dashboard/web/images/flink-logo.png similarity index 100% rename from flink-runtime-web/src/main/resources/web/images/flink-logo.png rename to flink-runtime-web/web-dashboard/web/images/flink-logo.png diff --git a/flink-runtime-web/src/main/resources/web/index.html b/flink-runtime-web/web-dashboard/web/index.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/index.html rename to flink-runtime-web/web-dashboard/web/index.html diff --git a/flink-runtime-web/src/main/resources/web/js/index.js b/flink-runtime-web/web-dashboard/web/js/index.js similarity index 100% rename from flink-runtime-web/src/main/resources/web/js/index.js rename to flink-runtime-web/web-dashboard/web/js/index.js diff --git a/flink-runtime-web/src/main/resources/web/js/vendor.js b/flink-runtime-web/web-dashboard/web/js/vendor.js similarity index 100% rename from flink-runtime-web/src/main/resources/web/js/vendor.js rename to flink-runtime-web/web-dashboard/web/js/vendor.js diff --git a/flink-runtime-web/src/main/resources/web/partials/jobmanager/config.html b/flink-runtime-web/web-dashboard/web/partials/jobmanager/config.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobmanager/config.html rename to flink-runtime-web/web-dashboard/web/partials/jobmanager/config.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobmanager/index.html b/flink-runtime-web/web-dashboard/web/partials/jobmanager/index.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobmanager/index.html rename to flink-runtime-web/web-dashboard/web/partials/jobmanager/index.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobmanager/stdout.html b/flink-runtime-web/web-dashboard/web/partials/jobmanager/stdout.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobmanager/stdout.html rename to flink-runtime-web/web-dashboard/web/partials/jobmanager/stdout.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/completed-jobs.html b/flink-runtime-web/web-dashboard/web/partials/jobs/completed-jobs.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/completed-jobs.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/completed-jobs.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.config.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.config.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.config.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.config.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.exceptions.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.exceptions.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.exceptions.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.exceptions.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.node-list.accumulators.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.node-list.accumulators.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.node-list.accumulators.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.node-list.accumulators.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.node-list.overview.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.node-list.overview.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.node-list.overview.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.node-list.overview.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.node.accumulators.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.node.accumulators.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.node.accumulators.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.node.accumulators.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.node.subtasks.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.node.subtasks.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.plan.node.subtasks.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.plan.node.subtasks.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.properties.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.properties.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.properties.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.properties.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.statistics.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.statistics.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.statistics.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.statistics.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.timeline.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.timeline.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.timeline.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.timeline.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/job.timeline.vertex.html b/flink-runtime-web/web-dashboard/web/partials/jobs/job.timeline.vertex.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/job.timeline.vertex.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/job.timeline.vertex.html diff --git a/flink-runtime-web/src/main/resources/web/partials/jobs/running-jobs.html b/flink-runtime-web/web-dashboard/web/partials/jobs/running-jobs.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/jobs/running-jobs.html rename to flink-runtime-web/web-dashboard/web/partials/jobs/running-jobs.html diff --git a/flink-runtime-web/src/main/resources/web/partials/overview.html b/flink-runtime-web/web-dashboard/web/partials/overview.html similarity index 100% rename from flink-runtime-web/src/main/resources/web/partials/overview.html rename to flink-runtime-web/web-dashboard/web/partials/overview.html diff --git a/flink-runtime-web/web-dashboard/web/partials/taskmanager/index.html b/flink-runtime-web/web-dashboard/web/partials/taskmanager/index.html new file mode 100644 index 0000000000000000000000000000000000000000..bf374091118bf8af6e83ac5523db285a1602ada4 --- /dev/null +++ b/flink-runtime-web/web-dashboard/web/partials/taskmanager/index.html @@ -0,0 +1,57 @@ + + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + +
Path, IDData PortLast HeartbeatAll SlotsFree SlotsCPU CoresPhysical MemoryFree MemoryFlink Managed Memory
+ {{ manager.path }} + +
{{ manager.id }}
+
{{ manager.dataPort }}{{ manager.timeSinceLastHeartbeat | amDateFormat:'YYYY-MM-DD, H:mm:ss' }}{{ manager.slotsNumber }}{{ manager.freeSlots }}{{ manager.cpuCores }}{{ manager.physicalMemory | bytes:MB }}{{ manager.freeMemory | bytes:MB }}{{ manager.managedMemory | bytes:MB }}
+
\ No newline at end of file diff --git a/flink-runtime/src/main/java/org/apache/flink/runtime/blob/FileSystemBlobStore.java b/flink-runtime/src/main/java/org/apache/flink/runtime/blob/FileSystemBlobStore.java index 8a037ad821a6959b5b362a72f9ad31ff8f6295f9..4351eb1f0c346def9b058d1cf69715c57480f315 100644 --- a/flink-runtime/src/main/java/org/apache/flink/runtime/blob/FileSystemBlobStore.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/blob/FileSystemBlobStore.java @@ -25,7 +25,6 @@ import org.apache.flink.configuration.Configuration; import org.apache.flink.configuration.IllegalConfigurationException; import org.apache.flink.core.fs.FileSystem; import org.apache.flink.core.fs.Path; -import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.runtime.util.IOUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -51,40 +50,27 @@ class FileSystemBlobStore implements BlobStore { private final String basePath; FileSystemBlobStore(Configuration config) throws IOException { - StateBackend stateBackend = StateBackend.fromConfig(config); + String stateBackendBasePath = config.getString( + ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, ""); - if (stateBackend == StateBackend.FILESYSTEM) { - String stateBackendBasePath = config.getString( - ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, ""); - - if (stateBackendBasePath.equals("")) { - throw new IllegalConfigurationException(String.format("Missing configuration for " + - "file system state backend recovery path. Please specify via " + - "'%s' key.", ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH)); - } + if (stateBackendBasePath.equals("")) { + throw new IllegalConfigurationException(String.format("Missing configuration for " + + "file system state backend recovery path. Please specify via " + + "'%s' key.", ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH)); + } - stateBackendBasePath += "/blob"; + stateBackendBasePath += "/blob"; - this.basePath = stateBackendBasePath; + this.basePath = stateBackendBasePath; - try { - FileSystem.get(new URI(basePath)).mkdirs(new Path(basePath)); - } - catch (URISyntaxException e) { - throw new IOException(e); - } - - LOG.info("Created blob directory {}.", basePath); + try { + FileSystem.get(new URI(basePath)).mkdirs(new Path(basePath)); } - else { - // Nothing else support at the moment - throw new IllegalConfigurationException( - String.format("Illegal state backend " + - "configuration '%s'. Please configure '%s' as state " + - "backend and specify the recovery path via '%s' key.", - stateBackend, StateBackend.FILESYSTEM, - ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH)); + catch (URISyntaxException e) { + throw new IOException(e); } + + LOG.info("Created blob directory {}.", basePath); } // - Put ------------------------------------------------------------------ diff --git a/flink-runtime/src/main/java/org/apache/flink/runtime/checkpoint/ZooKeeperCompletedCheckpointStore.java b/flink-runtime/src/main/java/org/apache/flink/runtime/checkpoint/ZooKeeperCompletedCheckpointStore.java index 62ab440ece0e54e03a9b2cc7c027a438b35848a0..cb2be648ed907ea80bf0cf81c40fb1c83294737c 100644 --- a/flink-runtime/src/main/java/org/apache/flink/runtime/checkpoint/ZooKeeperCompletedCheckpointStore.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/checkpoint/ZooKeeperCompletedCheckpointStore.java @@ -26,7 +26,7 @@ import org.apache.curator.utils.ZKPaths; import org.apache.flink.api.java.tuple.Tuple2; import org.apache.flink.runtime.jobmanager.RecoveryMode; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.runtime.state.StateHandleProvider; +import org.apache.flink.runtime.zookeeper.StateStorageHelper; import org.apache.flink.runtime.zookeeper.ZooKeeperStateHandleStore; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -92,7 +92,8 @@ public class ZooKeeperCompletedCheckpointStore implements CompletedCheckpointSto * @param client The Curator ZooKeeper client * @param checkpointsPath The ZooKeeper path for the checkpoints (needs to * start with a '/') - * @param stateHandleProvider The state handle provider for checkpoints + * @param stateStorage State storage to be used to persist the completed + * checkpoint * @throws Exception */ public ZooKeeperCompletedCheckpointStore( @@ -100,16 +101,16 @@ public class ZooKeeperCompletedCheckpointStore implements CompletedCheckpointSto ClassLoader userClassLoader, CuratorFramework client, String checkpointsPath, - StateHandleProvider stateHandleProvider) throws Exception { + StateStorageHelper stateStorage) throws Exception { checkArgument(maxNumberOfCheckpointsToRetain >= 1, "Must retain at least one checkpoint."); + checkNotNull(stateStorage, "State storage"); this.maxNumberOfCheckpointsToRetain = maxNumberOfCheckpointsToRetain; this.userClassLoader = checkNotNull(userClassLoader, "User class loader"); checkNotNull(client, "Curator client"); checkNotNull(checkpointsPath, "Checkpoints path"); - checkNotNull(stateHandleProvider, "State handle provider"); // Ensure that the checkpoints path exists client.newNamespaceAwareEnsurePath(checkpointsPath) @@ -118,8 +119,7 @@ public class ZooKeeperCompletedCheckpointStore implements CompletedCheckpointSto // All operations will have the path as root this.client = client.usingNamespace(client.getNamespace() + checkpointsPath); - this.checkpointsInZooKeeper = new ZooKeeperStateHandleStore<>( - this.client, stateHandleProvider); + this.checkpointsInZooKeeper = new ZooKeeperStateHandleStore<>(this.client, stateStorage); this.checkpointStateHandles = new ArrayDeque<>(maxNumberOfCheckpointsToRetain + 1); diff --git a/flink-runtime/src/main/java/org/apache/flink/runtime/jobmanager/ZooKeeperSubmittedJobGraphStore.java b/flink-runtime/src/main/java/org/apache/flink/runtime/jobmanager/ZooKeeperSubmittedJobGraphStore.java index 660f8bc852eaa534ee2e509bbb603e3fa086ee4a..a9ac77a603723dfb4beacb2ae00618ad1bac0a21 100644 --- a/flink-runtime/src/main/java/org/apache/flink/runtime/jobmanager/ZooKeeperSubmittedJobGraphStore.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/jobmanager/ZooKeeperSubmittedJobGraphStore.java @@ -26,7 +26,7 @@ import org.apache.curator.utils.ZKPaths; import org.apache.flink.api.common.JobID; import org.apache.flink.api.java.tuple.Tuple2; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.runtime.state.StateHandleProvider; +import org.apache.flink.runtime.zookeeper.StateStorageHelper; import org.apache.flink.runtime.zookeeper.ZooKeeperStateHandleStore; import org.apache.zookeeper.KeeperException; import org.slf4j.Logger; @@ -87,13 +87,21 @@ public class ZooKeeperSubmittedJobGraphStore implements SubmittedJobGraphStore { /** Flag indicating whether this instance is running. */ private boolean isRunning; + /** + * Submitted job graph store backed by ZooKeeper + * + * @param client ZooKeeper client + * @param currentJobsPath ZooKeeper path for current job graphs + * @param stateStorage State storage used to persist the submitted jobs + * @throws Exception + */ public ZooKeeperSubmittedJobGraphStore( CuratorFramework client, String currentJobsPath, - StateHandleProvider stateHandleProvider) throws Exception { + StateStorageHelper stateStorage) throws Exception { checkNotNull(currentJobsPath, "Current jobs path"); - checkNotNull(stateHandleProvider, "State handle provider"); + checkNotNull(stateStorage, "State storage"); // Keep a reference to the original client and not the namespace facade. The namespace // facade cannot be closed. @@ -104,11 +112,11 @@ public class ZooKeeperSubmittedJobGraphStore implements SubmittedJobGraphStore { .ensure(client.getZookeeperClient()); // All operations will have the path as root - client = client.usingNamespace(client.getNamespace() + currentJobsPath); + CuratorFramework facade = client.usingNamespace(client.getNamespace() + currentJobsPath); - this.jobGraphsInZooKeeper = new ZooKeeperStateHandleStore<>(client, stateHandleProvider); + this.jobGraphsInZooKeeper = new ZooKeeperStateHandleStore<>(facade, stateStorage); - this.pathCache = new PathChildrenCache(client, "/", false); + this.pathCache = new PathChildrenCache(facade, "/", false); pathCache.getListenable().addListener(new SubmittedJobGraphsPathCacheListener()); } diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/AbstractHeapKvState.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/AbstractHeapKvState.java similarity index 98% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/AbstractHeapKvState.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/AbstractHeapKvState.java index b974674a44fff1518be82f7e4cc76698dcd7a529..12250b97b9a5d1e737ce21d4ab026cb0bb0d8d90 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/AbstractHeapKvState.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/AbstractHeapKvState.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state; +package org.apache.flink.runtime.state; import org.apache.flink.api.common.typeutils.TypeSerializer; import org.apache.flink.core.memory.DataOutputView; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/KvState.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/KvState.java similarity index 98% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/KvState.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/KvState.java index 9c628f8197cdf7fd5536ddc8a0fb2f41c54e4c14..5cc16a7da801f9bb550b067dd8dae3345e44fde1 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/KvState.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/KvState.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state; +package org.apache.flink.runtime.state; import org.apache.flink.api.common.state.OperatorState; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/KvStateSnapshot.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/KvStateSnapshot.java similarity index 98% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/KvStateSnapshot.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/KvStateSnapshot.java index 6aa7a1e78f036b0b67af57cae975989683aab4ba..3d6c56c806ab7c223c1081f726429f5d3afb7a0b 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/KvStateSnapshot.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/KvStateSnapshot.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state; +package org.apache.flink.runtime.state; import org.apache.flink.api.common.typeutils.TypeSerializer; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/SerializedCheckpointData.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/SerializedCheckpointData.java similarity index 99% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/SerializedCheckpointData.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/SerializedCheckpointData.java index 2bbb4e2a76d2275dacb53ad2643e82fd83b7a09d..16ad3fde602b1b66d1a3544aaf2c41598d1142ff 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/SerializedCheckpointData.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/SerializedCheckpointData.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state; +package org.apache.flink.runtime.state; import org.apache.flink.api.common.typeutils.TypeSerializer; import org.apache.flink.api.java.tuple.Tuple2; diff --git a/flink-runtime/src/main/java/org/apache/flink/runtime/state/StateBackend.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/StateBackend.java index 7aa1ccfe329e7b6c00918ca6e8604a3995c7c3a1..f8b1cfdbed6420e81994811d78a7f9d77c810d40 100644 --- a/flink-runtime/src/main/java/org/apache/flink/runtime/state/StateBackend.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/StateBackend.java @@ -18,22 +18,196 @@ package org.apache.flink.runtime.state; -import org.apache.flink.configuration.ConfigConstants; -import org.apache.flink.configuration.Configuration; +import org.apache.flink.api.common.JobID; +import org.apache.flink.api.common.typeutils.TypeSerializer; +import org.apache.flink.core.memory.DataInputView; +import org.apache.flink.core.memory.DataInputViewStreamWrapper; +import org.apache.flink.core.memory.DataOutputView; +import org.apache.flink.core.memory.DataOutputViewStreamWrapper; -public enum StateBackend { - JOBMANAGER, FILESYSTEM; +import java.io.IOException; +import java.io.OutputStream; +import java.io.Serializable; + +/** + * A state backend defines how state is stored and snapshotted during checkpoints. + * + * @param The type of backend itself. This generic parameter is used to refer to the + * type of backend when creating state backed by this backend. + */ +public abstract class StateBackend> implements java.io.Serializable { + + private static final long serialVersionUID = 4620413814639220247L; + + // ------------------------------------------------------------------------ + // initialization and cleanup + // ------------------------------------------------------------------------ + + /** + * This method is called by the task upon deployment to initialize the state backend for + * data for a specific job. + * + * @param job The ID of the job for which the state backend instance checkpoints data. + * @throws Exception Overwritten versions of this method may throw exceptions, in which + * case the job that uses the state backend is considered failed during + * deployment. + */ + public abstract void initializeForJob(JobID job) throws Exception; + + /** + * Disposes all state associated with the current job. + * + * @throws Exception Exceptions may occur during disposal of the state and should be forwarded. + */ + public abstract void disposeAllStateForCurrentJob() throws Exception; + + /** + * Closes the state backend, releasing all internal resources, but does not delete any persistent + * checkpoint data. + * + * @throws Exception Exceptions can be forwarded and will be logged by the system + */ + public abstract void close() throws Exception; + + // ------------------------------------------------------------------------ + // key/value state + // ------------------------------------------------------------------------ /** - * Returns the configured {@link StateBackend}. + * Creates a key/value state backed by this state backend. + * + * @param keySerializer The serializer for the key. + * @param valueSerializer The serializer for the value. + * @param defaultValue The value that is returned when no other value has been associated with a key, yet. + * @param The type of the key. + * @param The type of the value. + * + * @return A new key/value state backed by this backend. + * + * @throws Exception Exceptions may occur during initialization of the state and should be forwarded. + */ + public abstract KvState createKvState( + TypeSerializer keySerializer, TypeSerializer valueSerializer, + V defaultValue) throws Exception; + + + // ------------------------------------------------------------------------ + // storing state for a checkpoint + // ------------------------------------------------------------------------ + + /** + * Creates an output stream that writes into the state of the given checkpoint. When the stream + * is closes, it returns a state handle that can retrieve the state back. + * + * @param checkpointID The ID of the checkpoint. + * @param timestamp The timestamp of the checkpoint. + * @return An output stream that writes state for the given checkpoint. + * + * @throws Exception Exceptions may occur while creating the stream and should be forwarded. + */ + public abstract CheckpointStateOutputStream createCheckpointStateOutputStream( + long checkpointID, long timestamp) throws Exception; + + /** + * Creates a {@link DataOutputView} stream that writes into the state of the given checkpoint. + * When the stream is closes, it returns a state handle that can retrieve the state back. * - * @param config The config to parse - * @return Configured state backend or {@link ConfigConstants#DEFAULT_RECOVERY_MODE} if not - * configured. - */ - public static StateBackend fromConfig(Configuration config) { - return StateBackend.valueOf(config.getString( - ConfigConstants.STATE_BACKEND, - ConfigConstants.DEFAULT_STATE_BACKEND).toUpperCase()); + * @param checkpointID The ID of the checkpoint. + * @param timestamp The timestamp of the checkpoint. + * @return An DataOutputView stream that writes state for the given checkpoint. + * + * @throws Exception Exceptions may occur while creating the stream and should be forwarded. + */ + public CheckpointStateOutputView createCheckpointStateOutputView( + long checkpointID, long timestamp) throws Exception { + return new CheckpointStateOutputView(createCheckpointStateOutputStream(checkpointID, timestamp)); + } + + /** + * Writes the given state into the checkpoint, and returns a handle that can retrieve the state back. + * + * @param state The state to be checkpointed. + * @param checkpointID The ID of the checkpoint. + * @param timestamp The timestamp of the checkpoint. + * @param The type of the state. + * + * @return A state handle that can retrieve the checkpoined state. + * + * @throws Exception Exceptions may occur during serialization / storing the state and should be forwarded. + */ + public abstract StateHandle checkpointStateSerializable( + S state, long checkpointID, long timestamp) throws Exception; + + + // ------------------------------------------------------------------------ + // Checkpoint state output stream + // ------------------------------------------------------------------------ + + /** + * A dedicated output stream that produces a {@link StreamStateHandle} when closed. + */ + public static abstract class CheckpointStateOutputStream extends OutputStream { + + /** + * Closes the stream and gets a state handle that can create an input stream + * producing the data written to this stream. + * + * @return A state handle that can create an input stream producing the data written to this stream. + * @throws IOException Thrown, if the stream cannot be closed. + */ + public abstract StreamStateHandle closeAndGetHandle() throws IOException; + } + + /** + * A dedicated DataOutputView stream that produces a {@code StateHandle} when closed. + */ + public static final class CheckpointStateOutputView extends DataOutputViewStreamWrapper { + + private final CheckpointStateOutputStream out; + + public CheckpointStateOutputView(CheckpointStateOutputStream out) { + super(out); + this.out = out; + } + + /** + * Closes the stream and gets a state handle that can create a DataInputView. + * producing the data written to this stream. + * + * @return A state handle that can create an input stream producing the data written to this stream. + * @throws IOException Thrown, if the stream cannot be closed. + */ + public StateHandle closeAndGetHandle() throws IOException { + return new DataInputViewHandle(out.closeAndGetHandle()); + } + + @Override + public void close() throws IOException { + out.close(); + } + } + + /** + * Simple state handle that resolved a {@link DataInputView} from a StreamStateHandle. + */ + private static final class DataInputViewHandle implements StateHandle { + + private static final long serialVersionUID = 2891559813513532079L; + + private final StreamStateHandle stream; + + private DataInputViewHandle(StreamStateHandle stream) { + this.stream = stream; + } + + @Override + public DataInputView getState(ClassLoader userCodeClassLoader) throws Exception { + return new DataInputViewStreamWrapper(stream.getState(userCodeClassLoader)); + } + + @Override + public void discardState() throws Exception { + stream.discardState(); + } } } diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/StateBackendFactory.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/StateBackendFactory.java similarity index 97% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/StateBackendFactory.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/StateBackendFactory.java index ad87eaed232593368fc36a7e3aedf4a9ee499d2e..5b622ebe5e5333f1e1a43fa91f3b33b1489465e9 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/StateBackendFactory.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/StateBackendFactory.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state; +package org.apache.flink.runtime.state; import org.apache.flink.configuration.Configuration; diff --git a/flink-runtime/src/main/java/org/apache/flink/runtime/state/StateHandleProviderFactory.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/StateHandleProviderFactory.java deleted file mode 100644 index 0086ac6f5158fbd04bd78a3aab2d67e1aaec5b9e..0000000000000000000000000000000000000000 --- a/flink-runtime/src/main/java/org/apache/flink/runtime/state/StateHandleProviderFactory.java +++ /dev/null @@ -1,61 +0,0 @@ -/* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.apache.flink.runtime.state; - -import org.apache.flink.configuration.ConfigConstants; -import org.apache.flink.configuration.Configuration; -import org.apache.flink.configuration.IllegalConfigurationException; - -import java.io.Serializable; - -/** - * State handler provider factory. - * - *

This is going to be superseded soon. - */ -public class StateHandleProviderFactory { - - /** - * Creates a {@link org.apache.flink.runtime.state.FileStateHandle.FileStateHandleProvider} at - * the configured recovery path. - */ - public static StateHandleProvider createRecoveryFileStateHandleProvider( - Configuration config) { - - StateBackend stateBackend = StateBackend.fromConfig(config); - - if (stateBackend == StateBackend.FILESYSTEM) { - String recoveryPath = config.getString( - ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, ""); - - if (recoveryPath.equals("")) { - throw new IllegalConfigurationException("Missing recovery path. Specify via " + - "configuration key '" + ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH + "'."); - } - else { - return FileStateHandle.createProvider(recoveryPath); - } - } - else { - throw new IllegalConfigurationException("Unexpected state backend configuration " + - stateBackend); - } - } - -} diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/StreamStateHandle.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/StreamStateHandle.java similarity index 95% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/StreamStateHandle.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/StreamStateHandle.java index 0fa59529df07310c072ee5fb1baaecf5a4195018..32c601e613492c7cdbca5b4e7da54942941a1651 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/StreamStateHandle.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/StreamStateHandle.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state; +package org.apache.flink.runtime.state; import org.apache.flink.runtime.state.StateHandle; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/AbstractFileState.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/AbstractFileState.java similarity index 97% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/AbstractFileState.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/AbstractFileState.java index c4a376eab20058ebe8f309d1d3b6fb03b9aa41d9..d64e2c4c853776c7913bfedd3785cfc655b5be3d 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/AbstractFileState.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/AbstractFileState.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.filesystem; +package org.apache.flink.runtime.state.filesystem; import org.apache.flink.core.fs.FileSystem; import org.apache.flink.core.fs.Path; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FileSerializableStateHandle.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FileSerializableStateHandle.java similarity index 97% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FileSerializableStateHandle.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FileSerializableStateHandle.java index 9bf5ec1c1b5c9226a1a0d188d033d1ea3d5ae055..b7e7cd16632ef4464dca2ea78c3a8f471bcde3cd 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FileSerializableStateHandle.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FileSerializableStateHandle.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.filesystem; +package org.apache.flink.runtime.state.filesystem; import org.apache.flink.core.fs.FSDataInputStream; import org.apache.flink.core.fs.Path; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FileStreamStateHandle.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FileStreamStateHandle.java similarity index 92% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FileStreamStateHandle.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FileStreamStateHandle.java index 79512d76bfb7a310071300e13e79d8f0803ae7c9..f4681ea743c9158389c618bc0c476b043324f069 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FileStreamStateHandle.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FileStreamStateHandle.java @@ -16,10 +16,10 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.filesystem; +package org.apache.flink.runtime.state.filesystem; import org.apache.flink.core.fs.Path; -import org.apache.flink.streaming.api.state.StreamStateHandle; +import org.apache.flink.runtime.state.StreamStateHandle; import java.io.InputStream; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsHeapKvState.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsHeapKvState.java similarity index 96% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsHeapKvState.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsHeapKvState.java index 107a3be6ae6744d198d92d816d37264f1d31b5ee..e3116dddac1b99e7c986e3b9c2fc8591d51327c2 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsHeapKvState.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsHeapKvState.java @@ -16,11 +16,11 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.filesystem; +package org.apache.flink.runtime.state.filesystem; import org.apache.flink.api.common.typeutils.TypeSerializer; import org.apache.flink.core.memory.OutputViewDataOutputStreamWrapper; -import org.apache.flink.streaming.api.state.AbstractHeapKvState; +import org.apache.flink.runtime.state.AbstractHeapKvState; import java.io.DataOutputStream; import java.util.HashMap; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsHeapKvStateSnapshot.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsHeapKvStateSnapshot.java similarity index 96% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsHeapKvStateSnapshot.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsHeapKvStateSnapshot.java index c7117f8c1eab76590641aea98ea8116cba6938fb..781ee3dafd89d781f6709d1f3b20a401472e99f5 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsHeapKvStateSnapshot.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsHeapKvStateSnapshot.java @@ -16,13 +16,13 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.filesystem; +package org.apache.flink.runtime.state.filesystem; import org.apache.flink.api.common.typeutils.TypeSerializer; import org.apache.flink.core.fs.FSDataInputStream; import org.apache.flink.core.fs.Path; import org.apache.flink.core.memory.InputViewDataInputStreamWrapper; -import org.apache.flink.streaming.api.state.KvStateSnapshot; +import org.apache.flink.runtime.state.KvStateSnapshot; import java.io.DataInputStream; import java.util.HashMap; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsStateBackend.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsStateBackend.java similarity index 99% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsStateBackend.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsStateBackend.java index 3cbd2277b9ee0d6c2509b15bd26624cc0f4331c6..045c4113abff45b4d075905ce14e444a3905a5ae 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsStateBackend.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsStateBackend.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.filesystem; +package org.apache.flink.runtime.state.filesystem; import org.apache.flink.api.common.JobID; import org.apache.flink.api.common.typeutils.TypeSerializer; @@ -24,7 +24,7 @@ import org.apache.flink.core.fs.FSDataOutputStream; import org.apache.flink.core.fs.FileSystem; import org.apache.flink.core.fs.Path; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.streaming.api.state.StateBackend; +import org.apache.flink.runtime.state.StateBackend; import org.slf4j.Logger; import org.slf4j.LoggerFactory; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsStateBackendFactory.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsStateBackendFactory.java similarity index 89% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsStateBackendFactory.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsStateBackendFactory.java index f0ad6bd2a41468f7280782912ba62a91b586c3bf..e687f7f5eb1c86c21a2be34d5c330e224f985ddc 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/filesystem/FsStateBackendFactory.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/filesystem/FsStateBackendFactory.java @@ -16,15 +16,15 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.filesystem; +package org.apache.flink.runtime.state.filesystem; import org.apache.flink.configuration.Configuration; import org.apache.flink.configuration.IllegalConfigurationException; import org.apache.flink.core.fs.Path; -import org.apache.flink.streaming.api.state.StateBackendFactory; +import org.apache.flink.runtime.state.StateBackendFactory; /** - * A factory that creates an {@link org.apache.flink.streaming.api.state.filesystem.FsStateBackend} + * A factory that creates an {@link org.apache.flink.runtime.state.filesystem.FsStateBackend} * from a configuration. */ public class FsStateBackendFactory implements StateBackendFactory { diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/ByteStreamStateHandle.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/ByteStreamStateHandle.java similarity index 92% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/ByteStreamStateHandle.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/ByteStreamStateHandle.java index 7952e58e8bdc4bcdfcb6ae25c393371a909401ad..29762f7c1f9705608bb3cde65fe85a781c94b59b 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/ByteStreamStateHandle.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/ByteStreamStateHandle.java @@ -16,9 +16,9 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.memory; +package org.apache.flink.runtime.state.memory; -import org.apache.flink.streaming.api.state.StreamStateHandle; +import org.apache.flink.runtime.state.StreamStateHandle; import java.io.ByteArrayInputStream; import java.io.InputStream; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/MemHeapKvState.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/MemHeapKvState.java similarity index 94% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/MemHeapKvState.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/MemHeapKvState.java index e611887174b20756f339ce23d7c352cb2e8edff3..96cb440710fd680c2cdaa005dd595b204293b678 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/MemHeapKvState.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/MemHeapKvState.java @@ -16,11 +16,11 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.memory; +package org.apache.flink.runtime.state.memory; import org.apache.flink.api.common.typeutils.TypeSerializer; import org.apache.flink.runtime.util.DataOutputSerializer; -import org.apache.flink.streaming.api.state.AbstractHeapKvState; +import org.apache.flink.runtime.state.AbstractHeapKvState; import java.util.HashMap; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/MemoryHeapKvStateSnapshot.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/MemoryHeapKvStateSnapshot.java similarity index 96% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/MemoryHeapKvStateSnapshot.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/MemoryHeapKvStateSnapshot.java index 7f50379a8bfcbc89173c4b6b90a9f35fff7c18a7..1b03defb0085240ff2887e95510e23a8220da560 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/MemoryHeapKvStateSnapshot.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/MemoryHeapKvStateSnapshot.java @@ -16,11 +16,11 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.memory; +package org.apache.flink.runtime.state.memory; import org.apache.flink.api.common.typeutils.TypeSerializer; import org.apache.flink.runtime.util.DataInputDeserializer; -import org.apache.flink.streaming.api.state.KvStateSnapshot; +import org.apache.flink.runtime.state.KvStateSnapshot; import java.util.HashMap; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/MemoryStateBackend.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/MemoryStateBackend.java similarity index 97% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/MemoryStateBackend.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/MemoryStateBackend.java index 05368bd0b73bc4eaf11ece25917fa917073c5ffc..8d297d4902d7872592b2a8c7c0406aaf2c71c15a 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/MemoryStateBackend.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/MemoryStateBackend.java @@ -16,13 +16,13 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.memory; +package org.apache.flink.runtime.state.memory; import org.apache.flink.api.common.JobID; import org.apache.flink.api.common.typeutils.TypeSerializer; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.streaming.api.state.StateBackend; -import org.apache.flink.streaming.api.state.StreamStateHandle; +import org.apache.flink.runtime.state.StateBackend; +import org.apache.flink.runtime.state.StreamStateHandle; import java.io.ByteArrayOutputStream; import java.io.IOException; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/SerializedStateHandle.java b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/SerializedStateHandle.java similarity index 96% rename from flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/SerializedStateHandle.java rename to flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/SerializedStateHandle.java index 163caddde6c7d373785ee230704d5e74a17b5019..c488dc98a2a9a1e598389695cb5ebe407b551211 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/memory/SerializedStateHandle.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/state/memory/SerializedStateHandle.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state.memory; +package org.apache.flink.runtime.state.memory; import org.apache.flink.runtime.state.StateHandle; import org.apache.flink.util.SerializedValue; diff --git a/flink-runtime/src/main/java/org/apache/flink/runtime/util/ZooKeeperUtils.java b/flink-runtime/src/main/java/org/apache/flink/runtime/util/ZooKeeperUtils.java index 79b9b7e345c856c6ca89773840a874fce5edb2c5..a32fc65552bac50e74e361400b73f6b28cbede84 100644 --- a/flink-runtime/src/main/java/org/apache/flink/runtime/util/ZooKeeperUtils.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/util/ZooKeeperUtils.java @@ -34,11 +34,14 @@ import org.apache.flink.runtime.jobmanager.SubmittedJobGraph; import org.apache.flink.runtime.jobmanager.ZooKeeperSubmittedJobGraphStore; import org.apache.flink.runtime.leaderelection.ZooKeeperLeaderElectionService; import org.apache.flink.runtime.leaderretrieval.ZooKeeperLeaderRetrievalService; -import org.apache.flink.runtime.state.StateHandleProvider; -import org.apache.flink.runtime.state.StateHandleProviderFactory; +import org.apache.flink.runtime.zookeeper.StateStorageHelper; +import org.apache.flink.runtime.zookeeper.filesystem.FileSystemStateStorageHelper; import org.slf4j.Logger; import org.slf4j.LoggerFactory; +import java.io.IOException; +import java.io.Serializable; + import static com.google.common.base.Preconditions.checkNotNull; public class ZooKeeperUtils { @@ -170,7 +173,7 @@ public class ZooKeeperUtils { String latchPath = configuration.getString(ConfigConstants.ZOOKEEPER_LATCH_PATH, ConfigConstants.DEFAULT_ZOOKEEPER_LATCH_PATH); String leaderPath = configuration.getString(ConfigConstants.ZOOKEEPER_LEADER_PATH, - ConfigConstants.DEFAULT_ZOOKEEPER_LEADER_PATH); + ConfigConstants.DEFAULT_ZOOKEEPER_LEADER_PATH); return new ZooKeeperLeaderElectionService(client, latchPath, leaderPath); } @@ -188,8 +191,7 @@ public class ZooKeeperUtils { checkNotNull(configuration, "Configuration"); - StateHandleProvider stateHandleProvider = - StateHandleProviderFactory.createRecoveryFileStateHandleProvider(configuration); + StateStorageHelper stateStorage = createFileSystemStateStorage(configuration, "submittedJobGraph"); // ZooKeeper submitted jobs root dir String zooKeeperSubmittedJobsPath = configuration.getString( @@ -197,7 +199,7 @@ public class ZooKeeperUtils { ConfigConstants.DEFAULT_ZOOKEEPER_JOBGRAPHS_PATH); return new ZooKeeperSubmittedJobGraphStore( - client, zooKeeperSubmittedJobsPath, stateHandleProvider); + client, zooKeeperSubmittedJobsPath, stateStorage); } /** @@ -219,21 +221,23 @@ public class ZooKeeperUtils { checkNotNull(configuration, "Configuration"); - StateHandleProvider stateHandleProvider = - StateHandleProviderFactory.createRecoveryFileStateHandleProvider(configuration); + String checkpointsPath = configuration.getString( + ConfigConstants.ZOOKEEPER_CHECKPOINTS_PATH, + ConfigConstants.DEFAULT_ZOOKEEPER_CHECKPOINTS_PATH); - String completedCheckpointsPath = configuration.getString( - ConfigConstants.ZOOKEEPER_CHECKPOINTS_PATH, - ConfigConstants.DEFAULT_ZOOKEEPER_CHECKPOINTS_PATH); - completedCheckpointsPath += ZooKeeperSubmittedJobGraphStore.getPathForJob(jobId); + StateStorageHelper stateStorage = createFileSystemStateStorage( + configuration, + "completedCheckpoint"); + + checkpointsPath += ZooKeeperSubmittedJobGraphStore.getPathForJob(jobId); return new ZooKeeperCompletedCheckpointStore( maxNumberOfCheckpointsToRetain, userClassLoader, client, - completedCheckpointsPath, - stateHandleProvider); + checkpointsPath, + stateStorage); } /** @@ -258,6 +262,30 @@ public class ZooKeeperUtils { return new ZooKeeperCheckpointIDCounter(client, checkpointIdCounterPath); } + /** + * Creates a {@link FileSystemStateStorageHelper} instance. + * + * @param configuration {@link Configuration} object + * @param prefix Prefix for the created files + * @param Type of the state objects + * @return {@link FileSystemStateStorageHelper} instance + * @throws IOException + */ + private static FileSystemStateStorageHelper createFileSystemStateStorage( + Configuration configuration, + String prefix) throws IOException { + + String rootPath = configuration.getString( + ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, ""); + + if (rootPath.equals("")) { + throw new IllegalConfigurationException("Missing recovery path. Specify via " + + "configuration key '" + ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH + "'."); + } else { + return new FileSystemStateStorageHelper(rootPath, prefix); + } + } + /** * Private constructor to prevent instantiation. */ diff --git a/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/StateStorageHelper.java b/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/StateStorageHelper.java new file mode 100644 index 0000000000000000000000000000000000000000..d18cace277e5fa36ea0c318afe9fdf61064ff843 --- /dev/null +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/StateStorageHelper.java @@ -0,0 +1,41 @@ +/* + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.apache.flink.runtime.zookeeper; + +import org.apache.flink.runtime.state.StateHandle; + +import java.io.Serializable; + +/** + * State storage helper which is used by {@ZooKeeperStateHandleStore} to persiste state before + * the state handle is written to ZooKeeper. + * + * @param + */ +public interface StateStorageHelper { + + /** + * Stores the given state and returns a state handle to it. + * + * @param state State to be stored + * @return State handle to the stored state + * @throws Exception + */ + StateHandle store(T state) throws Exception; +} diff --git a/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/ZooKeeperStateHandleStore.java b/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/ZooKeeperStateHandleStore.java index 936fe1ba52b62b569837d9686363973e9798d5da..6073a399ba993b732b603548720847a34f7a97af 100644 --- a/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/ZooKeeperStateHandleStore.java +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/ZooKeeperStateHandleStore.java @@ -23,12 +23,14 @@ import org.apache.curator.framework.api.BackgroundCallback; import org.apache.curator.utils.ZKPaths; import org.apache.flink.api.java.tuple.Tuple2; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.runtime.state.StateHandleProvider; import org.apache.flink.util.InstantiationUtil; import org.apache.zookeeper.CreateMode; import org.apache.zookeeper.KeeperException; import org.apache.zookeeper.data.Stat; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import java.io.IOException; import java.io.Serializable; import java.util.ArrayList; import java.util.List; @@ -65,11 +67,12 @@ import static com.google.common.base.Preconditions.checkNotNull; */ public class ZooKeeperStateHandleStore { + public static Logger LOG = LoggerFactory.getLogger(ZooKeeperStateHandleStore.class); + /** Curator ZooKeeper client */ private final CuratorFramework client; - /** State handle provider */ - private final StateHandleProvider stateHandleProvider; + private final StateStorageHelper storage; /** * Creates a {@link ZooKeeperStateHandleStore}. @@ -78,14 +81,13 @@ public class ZooKeeperStateHandleStore { * expected that the client's namespace ensures that the root * path is exclusive for all state handles managed by this * instance, e.g. client.usingNamespace("/stateHandles") - * @param stateHandleProvider The state handle provider for the state */ public ZooKeeperStateHandleStore( - CuratorFramework client, - StateHandleProvider stateHandleProvider) { + CuratorFramework client, + StateStorageHelper storage) throws IOException { this.client = checkNotNull(client, "Curator client"); - this.stateHandleProvider = checkNotNull(stateHandleProvider, "State handle provider"); + this.storage = checkNotNull(storage, "State storage"); } /** @@ -112,12 +114,14 @@ public class ZooKeeperStateHandleStore { * @return Created {@link StateHandle} * @throws Exception If a ZooKeeper or state handle operation fails */ - public StateHandle add(String pathInZooKeeper, T state, CreateMode createMode) throws Exception { + public StateHandle add( + String pathInZooKeeper, + T state, + CreateMode createMode) throws Exception { checkNotNull(pathInZooKeeper, "Path in ZooKeeper"); checkNotNull(state, "State"); - // Create the state handle. Nothing persisted yet. - StateHandle stateHandle = stateHandleProvider.createStateHandle(state); + StateHandle stateHandle = storage.store(state); boolean success = false; @@ -159,7 +163,7 @@ public class ZooKeeperStateHandleStore { StateHandle oldStateHandle = get(pathInZooKeeper); - StateHandle stateHandle = stateHandleProvider.createStateHandle(state); + StateHandle stateHandle = storage.store(state); boolean success = false; diff --git a/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/filesystem/FileSystemStateStorageHelper.java b/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/filesystem/FileSystemStateStorageHelper.java new file mode 100644 index 0000000000000000000000000000000000000000..d6b69e4fd691e54d57ba85121a08e789acf7b286 --- /dev/null +++ b/flink-runtime/src/main/java/org/apache/flink/runtime/zookeeper/filesystem/FileSystemStateStorageHelper.java @@ -0,0 +1,86 @@ +/* + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.apache.flink.runtime.zookeeper.filesystem; + +import com.google.common.base.Preconditions; +import org.apache.flink.core.fs.FSDataOutputStream; +import org.apache.flink.core.fs.FileSystem; +import org.apache.flink.core.fs.Path; +import org.apache.flink.runtime.state.StateHandle; +import org.apache.flink.runtime.state.filesystem.FileSerializableStateHandle; +import org.apache.flink.runtime.util.FileUtils; +import org.apache.flink.runtime.zookeeper.StateStorageHelper; + +import java.io.IOException; +import java.io.ObjectOutputStream; +import java.io.Serializable; + +/** + * {@link StateStorageHelper} implementation which stores the state in the given filesystem path. + * + * @param + */ +public class FileSystemStateStorageHelper implements StateStorageHelper { + + private final Path rootPath; + + private final String prefix; + + private final FileSystem fs; + + public FileSystemStateStorageHelper(String rootPath, String prefix) throws IOException { + this(new Path(rootPath), prefix); + } + + public FileSystemStateStorageHelper(Path rootPath, String prefix) throws IOException { + this.rootPath = Preconditions.checkNotNull(rootPath, "Root path"); + this.prefix = Preconditions.checkNotNull(prefix, "Prefix"); + + fs = FileSystem.get(rootPath.toUri()); + } + + @Override + public StateHandle store(T state) throws Exception { + Exception latestException = null; + + for (int attempt = 0; attempt < 10; attempt++) { + Path filePath = getNewFilePath(); + FSDataOutputStream outStream; + try { + outStream = fs.create(filePath, false); + } + catch (Exception e) { + latestException = e; + continue; + } + + try(ObjectOutputStream os = new ObjectOutputStream(outStream)) { + os.writeObject(state); + } + + return new FileSerializableStateHandle<>(filePath); + } + + throw new Exception("Could not open output stream for state backend", latestException); + } + + private Path getNewFilePath() { + return new Path(rootPath, FileUtils.getRandomFilename(prefix)); + } +} diff --git a/flink-runtime/src/main/scala/org/apache/flink/runtime/jobmanager/JobManager.scala b/flink-runtime/src/main/scala/org/apache/flink/runtime/jobmanager/JobManager.scala index ebc0ea95ace80ce8c79f152fa373f475038ac472..d9b69ad4406b7425fb74186e287b61ba21e35e20 100644 --- a/flink-runtime/src/main/scala/org/apache/flink/runtime/jobmanager/JobManager.scala +++ b/flink-runtime/src/main/scala/org/apache/flink/runtime/jobmanager/JobManager.scala @@ -1542,30 +1542,25 @@ object JobManager { } } - val webMonitor: Option[WebMonitor] = - if (configuration.getInteger(ConfigConstants.JOB_MANAGER_WEB_PORT_KEY, 0) >= 0) { - val address = AkkaUtils.getAddress(jobManagerSystem) + val address = AkkaUtils.getAddress(jobManagerSystem) - configuration.setString(ConfigConstants.JOB_MANAGER_IPC_ADDRESS_KEY, address.host.get) - configuration.setInteger(ConfigConstants.JOB_MANAGER_IPC_PORT_KEY, address.port.get) + configuration.setString(ConfigConstants.JOB_MANAGER_IPC_ADDRESS_KEY, address.host.get) + configuration.setInteger(ConfigConstants.JOB_MANAGER_IPC_PORT_KEY, address.port.get) - // start the job manager web frontend - if (configuration.getBoolean(ConfigConstants.JOB_MANAGER_NEW_WEB_FRONTEND_KEY, false)) { - val leaderRetrievalService = LeaderRetrievalUtils - .createLeaderRetrievalService(configuration) + val webMonitor: Option[WebMonitor] = + if (configuration.getInteger(ConfigConstants.JOB_MANAGER_WEB_PORT_KEY, 0) >= 0) { + LOG.info("Starting JobManger web frontend") + val leaderRetrievalService = LeaderRetrievalUtils + .createLeaderRetrievalService(configuration) - LOG.info("Starting NEW JobManger web frontend") - // start the new web frontend. we need to load this dynamically - // because it is not in the same project/dependencies - Some(startWebRuntimeMonitor(configuration, leaderRetrievalService, jobManagerSystem)) - } - else { - LOG.info("Starting JobManger web frontend") + // start the web frontend. we need to load this dynamically + // because it is not in the same project/dependencies + val webServer = WebMonitorUtils.startWebRuntimeMonitor( + configuration, + leaderRetrievalService, + jobManagerSystem) - // The old web frontend does not work with recovery mode - val leaderRetrievalService = StandaloneUtils.createLeaderRetrievalService(configuration) - Some(new WebInfoServer(configuration, leaderRetrievalService, jobManagerSystem)) - } + Option(webServer) } else { None @@ -1624,16 +1619,8 @@ object JobManager { monitor => val jobManagerAkkaUrl = JobManager.getRemoteJobManagerAkkaURL(configuration) monitor.start(jobManagerAkkaUrl) - LOG.info("Starting JobManger web frontend") - // start the web frontend. we need to load this dynamically - // because it is not in the same project/dependencies - val webServer = WebMonitorUtils.startWebRuntimeMonitor( - configuration, - leaderRetrievalService, - jobManagerSystem) } - (jobManagerSystem, jobManager, archive, webMonitor) } catch { diff --git a/flink-runtime/src/test/java/org/apache/flink/runtime/checkpoint/ZooKeeperCompletedCheckpointStoreITCase.java b/flink-runtime/src/test/java/org/apache/flink/runtime/checkpoint/ZooKeeperCompletedCheckpointStoreITCase.java index 4c6ddfd6ab49609b657a125e5c4b97f2943a4a41..dc6f550a92a128413bd76be4945bf29e098c45c6 100644 --- a/flink-runtime/src/test/java/org/apache/flink/runtime/checkpoint/ZooKeeperCompletedCheckpointStoreITCase.java +++ b/flink-runtime/src/test/java/org/apache/flink/runtime/checkpoint/ZooKeeperCompletedCheckpointStoreITCase.java @@ -19,6 +19,8 @@ package org.apache.flink.runtime.checkpoint; import org.apache.flink.runtime.state.LocalStateHandle; +import org.apache.flink.runtime.state.StateHandle; +import org.apache.flink.runtime.zookeeper.StateStorageHelper; import org.apache.flink.runtime.zookeeper.ZooKeeperTestEnvironment; import org.junit.AfterClass; import org.junit.Before; @@ -56,8 +58,12 @@ public class ZooKeeperCompletedCheckpointStoreITCase extends CompletedCheckpoint ClassLoader userLoader) throws Exception { return new ZooKeeperCompletedCheckpointStore(maxNumberOfCheckpointsToRetain, userLoader, - ZooKeeper.createClient(), CheckpointsPath, new LocalStateHandle - .LocalStateHandleProvider()); + ZooKeeper.createClient(), CheckpointsPath, new StateStorageHelper() { + @Override + public StateHandle store(CompletedCheckpoint state) throws Exception { + return new LocalStateHandle<>(state); + } + }); } // --------------------------------------------------------------------------------------------- diff --git a/flink-runtime/src/test/java/org/apache/flink/runtime/execution/librarycache/BlobLibraryCacheRecoveryITCase.java b/flink-runtime/src/test/java/org/apache/flink/runtime/execution/librarycache/BlobLibraryCacheRecoveryITCase.java index 4df8afb2900890dbccabbd5f2df1b5be61572fcc..ea4195c3bf0072eaf43319d8df63e06acf007bf3 100644 --- a/flink-runtime/src/test/java/org/apache/flink/runtime/execution/librarycache/BlobLibraryCacheRecoveryITCase.java +++ b/flink-runtime/src/test/java/org/apache/flink/runtime/execution/librarycache/BlobLibraryCacheRecoveryITCase.java @@ -18,7 +18,6 @@ package org.apache.flink.runtime.execution.librarycache; -import org.apache.commons.io.FileUtils; import org.apache.flink.api.common.JobID; import org.apache.flink.configuration.ConfigConstants; import org.apache.flink.configuration.Configuration; @@ -28,9 +27,9 @@ import org.apache.flink.runtime.blob.BlobKey; import org.apache.flink.runtime.blob.BlobServer; import org.apache.flink.runtime.executiongraph.ExecutionAttemptID; import org.apache.flink.runtime.jobmanager.RecoveryMode; -import org.junit.After; -import org.junit.Before; +import org.junit.Rule; import org.junit.Test; +import org.junit.rules.TemporaryFolder; import java.io.File; import java.io.FileInputStream; @@ -46,23 +45,8 @@ import static org.junit.Assert.assertEquals; public class BlobLibraryCacheRecoveryITCase { - private File recoveryDir; - - @Before - public void setUp() throws Exception { - recoveryDir = new File(FileUtils.getTempDirectory(), "BlobRecoveryITCaseDir"); - if (!recoveryDir.exists() && !recoveryDir.mkdirs()) { - throw new IllegalStateException("Failed to create temp directory for test"); - } - } - - @After - public void cleanUp() throws Exception { - if (recoveryDir != null) { - FileUtils.deleteDirectory(recoveryDir); - } - } - + @Rule + public TemporaryFolder temporaryFolder = new TemporaryFolder(); /** * Tests that with {@link RecoveryMode#ZOOKEEPER} distributed JARs are recoverable from any * participating BlobLibraryCacheManager. @@ -81,7 +65,7 @@ public class BlobLibraryCacheRecoveryITCase { Configuration config = new Configuration(); config.setString(ConfigConstants.RECOVERY_MODE, "ZOOKEEPER"); config.setString(ConfigConstants.STATE_BACKEND, "FILESYSTEM"); - config.setString(ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, recoveryDir.getPath()); + config.setString(ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, temporaryFolder.getRoot().getAbsolutePath()); for (int i = 0; i < server.length; i++) { server[i] = new BlobServer(config); @@ -170,7 +154,7 @@ public class BlobLibraryCacheRecoveryITCase { } // Verify everything is clean - File[] recoveryFiles = recoveryDir.listFiles(); + File[] recoveryFiles = temporaryFolder.getRoot().listFiles(); assertEquals("Unclean state backend: " + Arrays.toString(recoveryFiles), 0, recoveryFiles.length); } } diff --git a/flink-runtime/src/test/java/org/apache/flink/runtime/jobmanager/ZooKeeperSubmittedJobGraphsStoreITCase.java b/flink-runtime/src/test/java/org/apache/flink/runtime/jobmanager/ZooKeeperSubmittedJobGraphsStoreITCase.java index 861a7134f4a24494708fdbf83e35945d48194610..356ba3687ac81bb7df545e94366f9459dc3440fd 100644 --- a/flink-runtime/src/test/java/org/apache/flink/runtime/jobmanager/ZooKeeperSubmittedJobGraphsStoreITCase.java +++ b/flink-runtime/src/test/java/org/apache/flink/runtime/jobmanager/ZooKeeperSubmittedJobGraphsStoreITCase.java @@ -24,7 +24,9 @@ import org.apache.flink.runtime.akka.ListeningBehaviour; import org.apache.flink.runtime.jobgraph.JobGraph; import org.apache.flink.runtime.jobgraph.JobVertex; import org.apache.flink.runtime.jobmanager.SubmittedJobGraphStore.SubmittedJobGraphListener; -import org.apache.flink.runtime.state.LocalStateHandle.LocalStateHandleProvider; +import org.apache.flink.runtime.state.LocalStateHandle; +import org.apache.flink.runtime.state.StateHandle; +import org.apache.flink.runtime.zookeeper.StateStorageHelper; import org.apache.flink.runtime.zookeeper.ZooKeeperTestEnvironment; import org.apache.flink.util.TestLogger; import org.junit.AfterClass; @@ -54,8 +56,13 @@ public class ZooKeeperSubmittedJobGraphsStoreITCase extends TestLogger { private final static ZooKeeperTestEnvironment ZooKeeper = new ZooKeeperTestEnvironment(1); - private final static LocalStateHandleProvider StateHandleProvider = - new LocalStateHandleProvider<>(); + private final static StateStorageHelper localStateStorage = new StateStorageHelper() { + @Override + public StateHandle store(SubmittedJobGraph state) throws Exception { + return new LocalStateHandle<>(state); + } + }; + @AfterClass public static void tearDown() throws Exception { @@ -72,8 +79,9 @@ public class ZooKeeperSubmittedJobGraphsStoreITCase extends TestLogger { @Test public void testPutAndRemoveJobGraph() throws Exception { ZooKeeperSubmittedJobGraphStore jobGraphs = new ZooKeeperSubmittedJobGraphStore( - ZooKeeper.createClient(), "/testPutAndRemoveJobGraph", - StateHandleProvider); + ZooKeeper.createClient(), + "/testPutAndRemoveJobGraph", + localStateStorage); try { SubmittedJobGraphListener listener = mock(SubmittedJobGraphListener.class); @@ -125,7 +133,7 @@ public class ZooKeeperSubmittedJobGraphsStoreITCase extends TestLogger { @Test public void testRecoverJobGraphs() throws Exception { ZooKeeperSubmittedJobGraphStore jobGraphs = new ZooKeeperSubmittedJobGraphStore( - ZooKeeper.createClient(), "/testRecoverJobGraphs", StateHandleProvider); + ZooKeeper.createClient(), "/testRecoverJobGraphs", localStateStorage); try { SubmittedJobGraphListener listener = mock(SubmittedJobGraphListener.class); @@ -175,10 +183,10 @@ public class ZooKeeperSubmittedJobGraphsStoreITCase extends TestLogger { try { jobGraphs = new ZooKeeperSubmittedJobGraphStore( - ZooKeeper.createClient(), "/testConcurrentAddJobGraph", StateHandleProvider); + ZooKeeper.createClient(), "/testConcurrentAddJobGraph", localStateStorage); otherJobGraphs = new ZooKeeperSubmittedJobGraphStore( - ZooKeeper.createClient(), "/testConcurrentAddJobGraph", StateHandleProvider); + ZooKeeper.createClient(), "/testConcurrentAddJobGraph", localStateStorage); SubmittedJobGraph jobGraph = createSubmittedJobGraph(new JobID(), 0); @@ -234,10 +242,10 @@ public class ZooKeeperSubmittedJobGraphsStoreITCase extends TestLogger { @Test(expected = IllegalStateException.class) public void testUpdateJobGraphYouDidNotGetOrAdd() throws Exception { ZooKeeperSubmittedJobGraphStore jobGraphs = new ZooKeeperSubmittedJobGraphStore( - ZooKeeper.createClient(), "/testUpdateJobGraphYouDidNotGetOrAdd", StateHandleProvider); + ZooKeeper.createClient(), "/testUpdateJobGraphYouDidNotGetOrAdd", localStateStorage); ZooKeeperSubmittedJobGraphStore otherJobGraphs = new ZooKeeperSubmittedJobGraphStore( - ZooKeeper.createClient(), "/testUpdateJobGraphYouDidNotGetOrAdd", StateHandleProvider); + ZooKeeper.createClient(), "/testUpdateJobGraphYouDidNotGetOrAdd", localStateStorage); jobGraphs.start(null); otherJobGraphs.start(null); diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/api/state/FileStateBackendTest.java b/flink-runtime/src/test/java/org/apache/flink/runtime/state/FileStateBackendTest.java similarity index 97% rename from flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/api/state/FileStateBackendTest.java rename to flink-runtime/src/test/java/org/apache/flink/runtime/state/FileStateBackendTest.java index 73100d1ab34646de7e7b3a78d63e7a2ddfb79c34..481fb98cf8915d0865c38ebf8f48e5d7cda66589 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/api/state/FileStateBackendTest.java +++ b/flink-runtime/src/test/java/org/apache/flink/runtime/state/FileStateBackendTest.java @@ -16,7 +16,7 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state; +package org.apache.flink.runtime.state; import org.apache.commons.io.FileUtils; import org.apache.flink.api.common.JobID; @@ -28,9 +28,13 @@ import org.apache.flink.api.java.typeutils.runtime.ValueSerializer; import org.apache.flink.configuration.ConfigConstants; import org.apache.flink.core.fs.Path; import org.apache.flink.core.testutils.CommonTestUtils; +import org.apache.flink.runtime.state.KvState; +import org.apache.flink.runtime.state.KvStateSnapshot; +import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.streaming.api.state.filesystem.FileStreamStateHandle; -import org.apache.flink.streaming.api.state.filesystem.FsStateBackend; +import org.apache.flink.runtime.state.StreamStateHandle; +import org.apache.flink.runtime.state.filesystem.FileStreamStateHandle; +import org.apache.flink.runtime.state.filesystem.FsStateBackend; import org.apache.flink.types.StringValue; import org.apache.flink.util.OperatingSystem; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/api/state/MemoryStateBackendTest.java b/flink-runtime/src/test/java/org/apache/flink/runtime/state/MemoryStateBackendTest.java similarity index 95% rename from flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/api/state/MemoryStateBackendTest.java rename to flink-runtime/src/test/java/org/apache/flink/runtime/state/MemoryStateBackendTest.java index 3410d09a4c7702dc5d345c9a717bf6115a141203..5f95b33946aa15d90bf020d486a5ede67512cc38 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/api/state/MemoryStateBackendTest.java +++ b/flink-runtime/src/test/java/org/apache/flink/runtime/state/MemoryStateBackendTest.java @@ -16,15 +16,19 @@ * limitations under the License. */ -package org.apache.flink.streaming.api.state; +package org.apache.flink.runtime.state; import org.apache.flink.api.common.typeutils.TypeSerializer; import org.apache.flink.api.common.typeutils.base.FloatSerializer; import org.apache.flink.api.common.typeutils.base.IntSerializer; import org.apache.flink.api.common.typeutils.base.StringSerializer; import org.apache.flink.api.java.typeutils.runtime.ValueSerializer; +import org.apache.flink.runtime.state.KvState; +import org.apache.flink.runtime.state.KvStateSnapshot; +import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.streaming.api.state.memory.MemoryStateBackend; +import org.apache.flink.runtime.state.StreamStateHandle; +import org.apache.flink.runtime.state.memory.MemoryStateBackend; import org.apache.flink.types.StringValue; import org.junit.Test; @@ -36,7 +40,7 @@ import java.util.HashMap; import static org.junit.Assert.*; /** - * Tests for the {@link org.apache.flink.streaming.api.state.memory.MemoryStateBackend}. + * Tests for the {@link org.apache.flink.runtime.state.memory.MemoryStateBackend}. */ public class MemoryStateBackendTest { diff --git a/flink-runtime/src/test/java/org/apache/flink/runtime/testutils/ZooKeeperTestUtils.java b/flink-runtime/src/test/java/org/apache/flink/runtime/testutils/ZooKeeperTestUtils.java index d2e5b6ad4d17e20b9a8f16793b31f1a0ced3eddc..a65ec01457b69ae11dbe8ddc1fd7e71eff8d6dd8 100644 --- a/flink-runtime/src/test/java/org/apache/flink/runtime/testutils/ZooKeeperTestUtils.java +++ b/flink-runtime/src/test/java/org/apache/flink/runtime/testutils/ZooKeeperTestUtils.java @@ -21,6 +21,7 @@ package org.apache.flink.runtime.testutils; import org.apache.flink.configuration.ConfigConstants; import org.apache.flink.configuration.Configuration; import org.apache.flink.runtime.jobmanager.RecoveryMode; +import org.apache.flink.runtime.state.filesystem.FsStateBackendFactory; import static com.google.common.base.Preconditions.checkNotNull; @@ -79,7 +80,7 @@ public class ZooKeeperTestUtils { // File system state backend config.setString(ConfigConstants.STATE_BACKEND, "FILESYSTEM"); - config.setString(ConfigConstants.STATE_BACKEND_FS_DIR, fsStateHandlePath + "/checkpoints"); + config.setString(FsStateBackendFactory.CHECKPOINT_DIRECTORY_URI_CONF_KEY, fsStateHandlePath + "/checkpoints"); config.setString(ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, fsStateHandlePath + "/recovery"); // Akka failure detection and execution retries diff --git a/flink-runtime/src/test/java/org/apache/flink/runtime/zookeeper/ZooKeeperStateHandleStoreITCase.java b/flink-runtime/src/test/java/org/apache/flink/runtime/zookeeper/ZooKeeperStateHandleStoreITCase.java index f0130ec228e6aeedab446249c608648dd21a8763..788f70d777e0e9b45d466965ca5b8c2e56f6e9ae 100644 --- a/flink-runtime/src/test/java/org/apache/flink/runtime/zookeeper/ZooKeeperStateHandleStoreITCase.java +++ b/flink-runtime/src/test/java/org/apache/flink/runtime/zookeeper/ZooKeeperStateHandleStoreITCase.java @@ -23,7 +23,6 @@ import org.apache.curator.framework.api.BackgroundCallback; import org.apache.curator.framework.api.CuratorEvent; import org.apache.flink.api.java.tuple.Tuple2; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.runtime.state.StateHandleProvider; import org.apache.flink.util.InstantiationUtil; import org.apache.flink.util.TestLogger; import org.apache.zookeeper.CreateMode; @@ -83,11 +82,9 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { */ @Test public void testAdd() throws Exception { - // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); - - ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( - ZooKeeper.getClient(), stateHandleProvider); + LongStateStorage longStateStorage = new LongStateStorage(); + ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore( + ZooKeeper.getClient(), longStateStorage); // Config final String pathInZooKeeper = "/testAdd"; @@ -98,8 +95,8 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { // Verify // State handle created - assertEquals(1, stateHandleProvider.getStateHandles().size()); - assertEquals(state, stateHandleProvider.getStateHandles().get(0).getState(null)); + assertEquals(1, store.getAll().size()); + assertEquals(state, store.get(pathInZooKeeper).getState(null)); // Path created and is persistent Stat stat = ZooKeeper.getClient().checkExists().forPath(pathInZooKeeper); @@ -120,10 +117,9 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { */ @Test public void testAddWithCreateMode() throws Exception { - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); - - ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( - ZooKeeper.getClient(), stateHandleProvider); + LongStateStorage longStateStorage = new LongStateStorage(); + ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore( + ZooKeeper.getClient(), longStateStorage); // Config Long state = 3457347234L; @@ -151,8 +147,8 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { // Verify // State handle created - assertEquals(i + 1, stateHandleProvider.getStateHandles().size()); - assertEquals(state, stateHandleProvider.getStateHandles().get(i).getState(null)); + assertEquals(i + 1, store.getAll().size()); + assertEquals(state, longStateStorage.getStateHandles().get(i).getState(null)); // Path created Stat stat = ZooKeeper.getClient().checkExists().forPath(pathInZooKeeper); @@ -182,7 +178,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { */ @Test(expected = Exception.class) public void testAddAlreadyExistingPath() throws Exception { - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -198,7 +194,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testAddDiscardStateHandleAfterFailure() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); CuratorFramework client = spy(ZooKeeper.getClient()); when(client.create()).thenThrow(new RuntimeException("Expected test Exception.")); @@ -231,7 +227,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testReplace() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -270,10 +266,10 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { */ @Test(expected = Exception.class) public void testReplaceNonExistingPath() throws Exception { - StateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + StateStorageHelper stateStorage = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( - ZooKeeper.getClient(), stateHandleProvider); + ZooKeeper.getClient(), stateStorage); store.replace("/testReplaceNonExistingPath", 0, 1L); } @@ -284,7 +280,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testReplaceDiscardStateHandleAfterFailure() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); CuratorFramework client = spy(ZooKeeper.getClient()); when(client.setData()).thenThrow(new RuntimeException("Expected test Exception.")); @@ -329,7 +325,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testGetAndExists() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -354,7 +350,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { */ @Test(expected = Exception.class) public void testGetNonExistingPath() throws Exception { - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -368,7 +364,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testGetAll() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -399,7 +395,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testGetAllSortedByName() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -429,7 +425,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testRemove() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -453,7 +449,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testRemoveWithCallback() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -492,7 +488,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testRemoveAndDiscardState() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -514,7 +510,7 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { @Test public void testRemoveAndDiscardAllState() throws Exception { // Setup - LongStateHandleProvider stateHandleProvider = new LongStateHandleProvider(); + LongStateStorage stateHandleProvider = new LongStateStorage(); ZooKeeperStateHandleStore store = new ZooKeeperStateHandleStore<>( ZooKeeper.getClient(), stateHandleProvider); @@ -543,21 +539,19 @@ public class ZooKeeperStateHandleStoreITCase extends TestLogger { // Simple test helpers // --------------------------------------------------------------------------------------------- - private static class LongStateHandleProvider implements StateHandleProvider { - - private static final long serialVersionUID = 4572084854499402276L; + private static class LongStateStorage implements StateStorageHelper { private final List stateHandles = new ArrayList<>(); @Override - public StateHandle createStateHandle(Long state) { + public StateHandle store(Long state) throws Exception { LongStateHandle stateHandle = new LongStateHandle(state); stateHandles.add(stateHandle); return stateHandle; } - public List getStateHandles() { + List getStateHandles() { return stateHandles; } } diff --git a/flink-runtime/src/test/resources/log4j-test.properties b/flink-runtime/src/test/resources/log4j-test.properties index 1ca02aaa9b9efc9861f541e4c3c45ea02c36bd25..f77ed0731f1c65d99db5c9f16a9699c927539b8c 100644 --- a/flink-runtime/src/test/resources/log4j-test.properties +++ b/flink-runtime/src/test/resources/log4j-test.properties @@ -16,7 +16,7 @@ # limitations under the License. ################################################################################ -log4j.rootLogger=INFO, console +log4j.rootLogger=OFF, console # ----------------------------------------------------------------------------- # Console (use 'console') @@ -36,3 +36,4 @@ log4j.appender.file.layout.ConversionPattern=%d{HH:mm:ss,SSS} %-5p %-60c %x - %m # suppress the irrelevant (wrong) warnings from the netty channel handler log4j.logger.org.jboss.netty.channel.DefaultChannelPipeline=ERROR, console +log4j.logger.org.apache.flink.runtime.blob=DEBUG diff --git a/flink-staging/flink-fs-tests/src/test/java/org/apache/flink/hdfstests/FileStateBackendTest.java b/flink-staging/flink-fs-tests/src/test/java/org/apache/flink/hdfstests/FileStateBackendTest.java index 8b7fb1c4f6ad1ba37c9baaee6d28642afaf2a14c..4e4acd2b6fbe8edc8afb7579f285c83c3b1b13d7 100644 --- a/flink-staging/flink-fs-tests/src/test/java/org/apache/flink/hdfstests/FileStateBackendTest.java +++ b/flink-staging/flink-fs-tests/src/test/java/org/apache/flink/hdfstests/FileStateBackendTest.java @@ -27,11 +27,11 @@ import org.apache.flink.core.fs.FileSystem; import org.apache.flink.core.fs.Path; import org.apache.flink.core.testutils.CommonTestUtils; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.streaming.api.state.filesystem.FileStreamStateHandle; -import org.apache.flink.streaming.api.state.filesystem.FsStateBackend; +import org.apache.flink.runtime.state.filesystem.FileStreamStateHandle; +import org.apache.flink.runtime.state.filesystem.FsStateBackend; -import org.apache.flink.streaming.api.state.StateBackend; -import org.apache.flink.streaming.api.state.StreamStateHandle; +import org.apache.flink.runtime.state.StateBackend; +import org.apache.flink.runtime.state.StreamStateHandle; import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.hdfs.MiniDFSCluster; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/environment/StreamExecutionEnvironment.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/environment/StreamExecutionEnvironment.java index 26e1c9efcf357571dfae3da13ae3eea561254aac..98506e0dceb093747a1f7b6a77be77bb7d5bd4dc 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/environment/StreamExecutionEnvironment.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/environment/StreamExecutionEnvironment.java @@ -63,7 +63,7 @@ import org.apache.flink.streaming.api.functions.source.StatefulSequenceSource; import org.apache.flink.streaming.api.graph.StreamGraph; import org.apache.flink.streaming.api.graph.StreamGraphGenerator; import org.apache.flink.streaming.api.operators.StreamSource; -import org.apache.flink.streaming.api.state.StateBackend; +import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.streaming.api.transformations.StreamTransformation; import org.apache.flink.types.StringValue; import org.apache.flink.util.SplittableIterator; @@ -372,11 +372,11 @@ public abstract class StreamExecutionEnvironment { * the key/value state, and for checkpointed functions (implementing the interface * {@link org.apache.flink.streaming.api.checkpoint.Checkpointed}). * - *

The {@link org.apache.flink.streaming.api.state.memory.MemoryStateBackend} for example + *

The {@link org.apache.flink.runtime.state.memory.MemoryStateBackend} for example * maintains the state in heap memory, as objects. It is lightweight without extra dependencies, * but can checkpoint only small states (some counters). * - *

In contrast, the {@link org.apache.flink.streaming.api.state.filesystem.FsStateBackend} + *

In contrast, the {@link org.apache.flink.runtime.state.filesystem.FsStateBackend} * stores checkpoints of the state (also maintained as heap objects) in files. When using a replicated * file system (like HDFS, S3, MapR FS, Tachyon, etc) this will guarantee that state is not lost upon * failures of individual nodes and that streaming program can be executed highly available and strongly diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/functions/source/MessageAcknowledingSourceBase.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/functions/source/MessageAcknowledingSourceBase.java index 3817ede1afa9d6d9b3aa41155a009ec319c599cc..3ac63af71f17823b0a35353ec2ebbda33191c75d 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/functions/source/MessageAcknowledingSourceBase.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/functions/source/MessageAcknowledingSourceBase.java @@ -26,8 +26,7 @@ import org.apache.flink.api.java.typeutils.TypeExtractor; import org.apache.flink.configuration.Configuration; import org.apache.flink.streaming.api.checkpoint.CheckpointNotifier; import org.apache.flink.streaming.api.checkpoint.Checkpointed; -import org.apache.flink.streaming.api.functions.source.RichSourceFunction; -import org.apache.flink.streaming.api.state.SerializedCheckpointData; +import org.apache.flink.runtime.state.SerializedCheckpointData; import java.util.ArrayDeque; import java.util.ArrayList; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/graph/StreamConfig.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/graph/StreamConfig.java index 76be598cfc8f499a546e321a57a15c1e12b6d79b..11bf84fb79d3b91b61795e9363ac2938b917fc6f 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/graph/StreamConfig.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/graph/StreamConfig.java @@ -31,7 +31,7 @@ import org.apache.flink.runtime.util.ClassLoaderUtil; import org.apache.flink.streaming.api.CheckpointingMode; import org.apache.flink.streaming.api.collector.selector.OutputSelectorWrapper; import org.apache.flink.streaming.api.operators.StreamOperator; -import org.apache.flink.streaming.api.state.StateBackend; +import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.streaming.runtime.tasks.StreamTaskException; import org.apache.flink.util.InstantiationUtil; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/graph/StreamGraph.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/graph/StreamGraph.java index 06524060a55ef34c9c18ebfee47d18ba9db2fc27..be020d72b57b88c6caf8c69ecfb56e3df9a4176e 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/graph/StreamGraph.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/graph/StreamGraph.java @@ -50,7 +50,7 @@ import org.apache.flink.streaming.api.operators.OutputTypeConfigurable; import org.apache.flink.streaming.api.operators.StreamOperator; import org.apache.flink.streaming.api.operators.StreamSource; import org.apache.flink.streaming.api.operators.TwoInputStreamOperator; -import org.apache.flink.streaming.api.state.StateBackend; +import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.streaming.runtime.partitioner.ForwardPartitioner; import org.apache.flink.streaming.runtime.partitioner.RebalancePartitioner; import org.apache.flink.streaming.runtime.partitioner.StreamPartitioner; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/operators/AbstractStreamOperator.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/operators/AbstractStreamOperator.java index 9e60e9a0df8a9dd18d1f39cccd0b5a9ba3e31561..078679ddb4b9e05ac1e570511811b8c76818c2b5 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/operators/AbstractStreamOperator.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/operators/AbstractStreamOperator.java @@ -24,9 +24,9 @@ import org.apache.flink.api.common.typeinfo.TypeInformation; import org.apache.flink.api.common.typeutils.TypeSerializer; import org.apache.flink.api.java.functions.KeySelector; import org.apache.flink.streaming.api.graph.StreamConfig; -import org.apache.flink.streaming.api.state.KvState; -import org.apache.flink.streaming.api.state.KvStateSnapshot; -import org.apache.flink.streaming.api.state.StateBackend; +import org.apache.flink.runtime.state.KvState; +import org.apache.flink.runtime.state.KvStateSnapshot; +import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.streaming.runtime.operators.Triggerable; import org.apache.flink.streaming.runtime.streamrecord.StreamRecord; import org.apache.flink.streaming.runtime.tasks.StreamTask; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/operators/AbstractUdfStreamOperator.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/operators/AbstractUdfStreamOperator.java index a991fd3135e953733652d2f3d058d637e12f2ac5..17bd08d87d2b3adb8485d67544df7f040c2a6167 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/operators/AbstractUdfStreamOperator.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/operators/AbstractUdfStreamOperator.java @@ -27,7 +27,7 @@ import org.apache.flink.runtime.state.StateHandle; import org.apache.flink.streaming.api.checkpoint.CheckpointNotifier; import org.apache.flink.streaming.api.checkpoint.Checkpointed; import org.apache.flink.streaming.api.graph.StreamConfig; -import org.apache.flink.streaming.api.state.StateBackend; +import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.streaming.runtime.streamrecord.StreamRecord; import org.apache.flink.streaming.runtime.tasks.StreamTask; import org.apache.flink.streaming.runtime.tasks.StreamTaskState; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/StateBackend.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/StateBackend.java deleted file mode 100644 index f4391adc280ae71f0483300648c7e7361492bbc2..0000000000000000000000000000000000000000 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/api/state/StateBackend.java +++ /dev/null @@ -1,214 +0,0 @@ -/* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.apache.flink.streaming.api.state; - -import org.apache.flink.api.common.JobID; -import org.apache.flink.api.common.typeutils.TypeSerializer; -import org.apache.flink.core.memory.DataInputView; -import org.apache.flink.core.memory.DataInputViewStreamWrapper; -import org.apache.flink.core.memory.DataOutputView; -import org.apache.flink.core.memory.DataOutputViewStreamWrapper; -import org.apache.flink.runtime.state.StateHandle; - -import java.io.IOException; -import java.io.OutputStream; -import java.io.Serializable; - -/** - * A state backend defines how state is stored and snapshotted during checkpoints. - * - * @param The type of backend itself. This generic parameter is used to refer to the - * type of backend when creating state backed by this backend. - */ -public abstract class StateBackend> implements java.io.Serializable { - - private static final long serialVersionUID = 4620413814639220247L; - - // ------------------------------------------------------------------------ - // initialization and cleanup - // ------------------------------------------------------------------------ - - /** - * This method is called by the task upon deployment to initialize the state backend for - * data for a specific job. - * - * @param job The ID of the job for which the state backend instance checkpoints data. - * @throws Exception Overwritten versions of this method may throw exceptions, in which - * case the job that uses the state backend is considered failed during - * deployment. - */ - public abstract void initializeForJob(JobID job) throws Exception; - - /** - * Disposes all state associated with the current job. - * - * @throws Exception Exceptions may occur during disposal of the state and should be forwarded. - */ - public abstract void disposeAllStateForCurrentJob() throws Exception; - - /** - * Closes the state backend, releasing all internal resources, but does not delete any persistent - * checkpoint data. - * - * @throws Exception Exceptions can be forwarded and will be logged by the system - */ - public abstract void close() throws Exception; - - // ------------------------------------------------------------------------ - // key/value state - // ------------------------------------------------------------------------ - - /** - * Creates a key/value state backed by this state backend. - * - * @param keySerializer The serializer for the key. - * @param valueSerializer The serializer for the value. - * @param defaultValue The value that is returned when no other value has been associated with a key, yet. - * @param The type of the key. - * @param The type of the value. - * - * @return A new key/value state backed by this backend. - * - * @throws Exception Exceptions may occur during initialization of the state and should be forwarded. - */ - public abstract KvState createKvState( - TypeSerializer keySerializer, TypeSerializer valueSerializer, - V defaultValue) throws Exception; - - - // ------------------------------------------------------------------------ - // storing state for a checkpoint - // ------------------------------------------------------------------------ - - /** - * Creates an output stream that writes into the state of the given checkpoint. When the stream - * is closes, it returns a state handle that can retrieve the state back. - * - * @param checkpointID The ID of the checkpoint. - * @param timestamp The timestamp of the checkpoint. - * @return An output stream that writes state for the given checkpoint. - * - * @throws Exception Exceptions may occur while creating the stream and should be forwarded. - */ - public abstract CheckpointStateOutputStream createCheckpointStateOutputStream( - long checkpointID, long timestamp) throws Exception; - - /** - * Creates a {@link DataOutputView} stream that writes into the state of the given checkpoint. - * When the stream is closes, it returns a state handle that can retrieve the state back. - * - * @param checkpointID The ID of the checkpoint. - * @param timestamp The timestamp of the checkpoint. - * @return An DataOutputView stream that writes state for the given checkpoint. - * - * @throws Exception Exceptions may occur while creating the stream and should be forwarded. - */ - public CheckpointStateOutputView createCheckpointStateOutputView( - long checkpointID, long timestamp) throws Exception { - return new CheckpointStateOutputView(createCheckpointStateOutputStream(checkpointID, timestamp)); - } - - /** - * Writes the given state into the checkpoint, and returns a handle that can retrieve the state back. - * - * @param state The state to be checkpointed. - * @param checkpointID The ID of the checkpoint. - * @param timestamp The timestamp of the checkpoint. - * @param The type of the state. - * - * @return A state handle that can retrieve the checkpoined state. - * - * @throws Exception Exceptions may occur during serialization / storing the state and should be forwarded. - */ - public abstract StateHandle checkpointStateSerializable( - S state, long checkpointID, long timestamp) throws Exception; - - - // ------------------------------------------------------------------------ - // Checkpoint state output stream - // ------------------------------------------------------------------------ - - /** - * A dedicated output stream that produces a {@link StreamStateHandle} when closed. - */ - public static abstract class CheckpointStateOutputStream extends OutputStream { - - /** - * Closes the stream and gets a state handle that can create an input stream - * producing the data written to this stream. - * - * @return A state handle that can create an input stream producing the data written to this stream. - * @throws IOException Thrown, if the stream cannot be closed. - */ - public abstract StreamStateHandle closeAndGetHandle() throws IOException; - } - - /** - * A dedicated DataOutputView stream that produces a {@code StateHandle} when closed. - */ - public static final class CheckpointStateOutputView extends DataOutputViewStreamWrapper { - - private final CheckpointStateOutputStream out; - - public CheckpointStateOutputView(CheckpointStateOutputStream out) { - super(out); - this.out = out; - } - - /** - * Closes the stream and gets a state handle that can create a DataInputView. - * producing the data written to this stream. - * - * @return A state handle that can create an input stream producing the data written to this stream. - * @throws IOException Thrown, if the stream cannot be closed. - */ - public StateHandle closeAndGetHandle() throws IOException { - return new DataInputViewHandle(out.closeAndGetHandle()); - } - - @Override - public void close() throws IOException { - out.close(); - } - } - - /** - * Simple state handle that resolved a {@link DataInputView} from a StreamStateHandle. - */ - private static final class DataInputViewHandle implements StateHandle { - - private static final long serialVersionUID = 2891559813513532079L; - - private final StreamStateHandle stream; - - private DataInputViewHandle(StreamStateHandle stream) { - this.stream = stream; - } - - @Override - public DataInputView getState(ClassLoader userCodeClassLoader) throws Exception { - return new DataInputViewStreamWrapper(stream.getState(userCodeClassLoader)); - } - - @Override - public void discardState() throws Exception { - stream.discardState(); - } - } -} diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/operators/windowing/AbstractAlignedProcessingTimeWindowOperator.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/operators/windowing/AbstractAlignedProcessingTimeWindowOperator.java index cf8575e467af564b38296c61b1ed86f9cf4a0d4c..9964760b27e3436ffb813805f62d694b6b1ca747 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/operators/windowing/AbstractAlignedProcessingTimeWindowOperator.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/operators/windowing/AbstractAlignedProcessingTimeWindowOperator.java @@ -29,7 +29,7 @@ import org.apache.flink.runtime.util.MathUtils; import org.apache.flink.streaming.api.operators.AbstractUdfStreamOperator; import org.apache.flink.streaming.api.operators.OneInputStreamOperator; import org.apache.flink.streaming.api.operators.TimestampedCollector; -import org.apache.flink.streaming.api.state.StateBackend; +import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.streaming.api.watermark.Watermark; import org.apache.flink.streaming.api.windowing.windows.TimeWindow; import org.apache.flink.streaming.runtime.operators.Triggerable; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/tasks/StreamTask.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/tasks/StreamTask.java index 72a8c259768763a501848a52e2c8e857361de8d8..8c58e29f3dd4f5727e39aafca5d704d79eff13c9 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/tasks/StreamTask.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/tasks/StreamTask.java @@ -30,17 +30,16 @@ import org.apache.flink.runtime.accumulators.AccumulatorRegistry; import org.apache.flink.runtime.io.network.api.CheckpointBarrier; import org.apache.flink.runtime.jobgraph.tasks.AbstractInvokable; import org.apache.flink.runtime.jobgraph.tasks.StatefulTask; -import org.apache.flink.runtime.state.StateBackend; import org.apache.flink.runtime.taskmanager.DispatcherThreadFactory; import org.apache.flink.runtime.util.event.EventListener; import org.apache.flink.streaming.api.graph.StreamConfig; import org.apache.flink.streaming.api.operators.Output; import org.apache.flink.streaming.api.operators.StreamOperator; -import org.apache.flink.streaming.api.state.StateBackend; -import org.apache.flink.streaming.api.state.StateBackendFactory; -import org.apache.flink.streaming.api.state.filesystem.FsStateBackend; -import org.apache.flink.streaming.api.state.filesystem.FsStateBackendFactory; -import org.apache.flink.streaming.api.state.memory.MemoryStateBackend; +import org.apache.flink.runtime.state.StateBackend; +import org.apache.flink.runtime.state.StateBackendFactory; +import org.apache.flink.runtime.state.filesystem.FsStateBackend; +import org.apache.flink.runtime.state.filesystem.FsStateBackendFactory; +import org.apache.flink.runtime.state.memory.MemoryStateBackend; import org.apache.flink.streaming.runtime.io.RecordWriterOutput; import org.apache.flink.streaming.runtime.operators.Triggerable; import org.apache.flink.streaming.runtime.streamrecord.StreamRecord; @@ -493,55 +492,52 @@ public abstract class StreamTask> private StateBackend createStateBackend() throws Exception { StateBackend configuredBackend = configuration.getStateBackend(userClassLoader); - + if (configuredBackend != null) { // backend has been configured on the environment LOG.info("Using user-defined state backend: " + configuredBackend); return configuredBackend; - } - else { + } else { // see if we have a backend specified in the configuration Configuration flinkConfig = getEnvironment().getTaskManagerInfo().getConfiguration(); String backendName = flinkConfig.getString(ConfigConstants.STATE_BACKEND, null); - + if (backendName == null) { LOG.warn("No state backend has been specified, using default state backend (Memory / JobManager)"); backendName = "jobmanager"; } - + backendName = backendName.toLowerCase(); switch (backendName) { case "jobmanager": LOG.info("State backend is set to heap memory (checkpoint to jobmanager)"); return MemoryStateBackend.defaultInstance(); - + case "filesystem": FsStateBackend backend = new FsStateBackendFactory().createFromConfig(flinkConfig); LOG.info("State backend is set to heap memory (checkpoints to filesystem \"" - + backend.getBasePath() + "\")"); + + backend.getBasePath() + "\")"); return backend; - + default: try { @SuppressWarnings("rawtypes") Class clazz = - Class.forName(backendName, false, userClassLoader).asSubclass(StateBackendFactory.class); + Class.forName(backendName, false, userClassLoader).asSubclass(StateBackendFactory.class); return (StateBackend) clazz.newInstance(); - } - catch (ClassNotFoundException e) { + } catch (ClassNotFoundException e) { throw new IllegalConfigurationException("Cannot find configured state backend: " + backendName); - } - catch (ClassCastException e) { + } catch (ClassCastException e) { throw new IllegalConfigurationException("The class configured under '" + - ConfigConstants.STATE_BACKEND + "' is not a valid state backend factory (" + - backendName + ')'); - } - catch (Throwable t) { + ConfigConstants.STATE_BACKEND + "' is not a valid state backend factory (" + + backendName + ')'); + } catch (Throwable t) { throw new IllegalConfigurationException("Cannot create configured state backend", t); } } } + } /** * Registers a timer. diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/tasks/StreamTaskState.java b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/tasks/StreamTaskState.java index 334fd44417564367682d529a10a2448e1f80a42a..afeabd9ed6f5da922345e55d706d3177a3d89630 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/tasks/StreamTaskState.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/main/java/org/apache/flink/streaming/runtime/tasks/StreamTaskState.java @@ -19,7 +19,7 @@ package org.apache.flink.streaming.runtime.tasks; import org.apache.flink.runtime.state.StateHandle; -import org.apache.flink.streaming.api.state.KvStateSnapshot; +import org.apache.flink.runtime.state.KvStateSnapshot; import java.io.Serializable; import java.util.ConcurrentModificationException; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/runtime/operators/windowing/AccumulatingAlignedProcessingTimeWindowOperatorTest.java b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/runtime/operators/windowing/AccumulatingAlignedProcessingTimeWindowOperatorTest.java index dd76a67214ac3a3cd708d6199a4499852bab6003..ad3c838bb5f8013e609bef0aa10e2a39345f24be 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/runtime/operators/windowing/AccumulatingAlignedProcessingTimeWindowOperatorTest.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/runtime/operators/windowing/AccumulatingAlignedProcessingTimeWindowOperatorTest.java @@ -28,8 +28,8 @@ import org.apache.flink.runtime.execution.Environment; import org.apache.flink.streaming.api.functions.windowing.WindowFunction; import org.apache.flink.streaming.api.graph.StreamConfig; import org.apache.flink.streaming.api.operators.Output; -import org.apache.flink.streaming.api.state.StateBackend; -import org.apache.flink.streaming.api.state.memory.MemoryStateBackend; +import org.apache.flink.runtime.state.StateBackend; +import org.apache.flink.runtime.state.memory.MemoryStateBackend; import org.apache.flink.streaming.api.windowing.windows.TimeWindow; import org.apache.flink.streaming.runtime.operators.Triggerable; import org.apache.flink.streaming.runtime.streamrecord.StreamRecord; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/runtime/operators/windowing/AggregatingAlignedProcessingTimeWindowOperatorTest.java b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/runtime/operators/windowing/AggregatingAlignedProcessingTimeWindowOperatorTest.java index ab8e5510a27df1714c8b64128ce9738cad185444..4bd260fe877d5c63b6ff9bc7f57d66956152771f 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/runtime/operators/windowing/AggregatingAlignedProcessingTimeWindowOperatorTest.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/runtime/operators/windowing/AggregatingAlignedProcessingTimeWindowOperatorTest.java @@ -28,8 +28,8 @@ import org.apache.flink.configuration.Configuration; import org.apache.flink.runtime.execution.Environment; import org.apache.flink.streaming.api.graph.StreamConfig; import org.apache.flink.streaming.api.operators.Output; -import org.apache.flink.streaming.api.state.StateBackend; -import org.apache.flink.streaming.api.state.memory.MemoryStateBackend; +import org.apache.flink.runtime.state.StateBackend; +import org.apache.flink.runtime.state.memory.MemoryStateBackend; import org.apache.flink.streaming.runtime.operators.Triggerable; import org.apache.flink.streaming.runtime.streamrecord.StreamRecord; import org.apache.flink.streaming.runtime.tasks.StreamTask; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/MockContext.java b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/MockContext.java index 81d3a6970d711c235a160a2d313de6c772830fd2..0c708c656cae5956efeeb24b580f10634f8cc96b 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/MockContext.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/MockContext.java @@ -36,8 +36,8 @@ import org.apache.flink.runtime.operators.testutils.MockInputSplitProvider; import org.apache.flink.streaming.api.graph.StreamConfig; import org.apache.flink.streaming.api.operators.OneInputStreamOperator; import org.apache.flink.streaming.api.operators.Output; -import org.apache.flink.streaming.api.state.StateBackend; -import org.apache.flink.streaming.api.state.memory.MemoryStateBackend; +import org.apache.flink.runtime.state.StateBackend; +import org.apache.flink.runtime.state.memory.MemoryStateBackend; import org.apache.flink.streaming.runtime.operators.Triggerable; import org.apache.flink.streaming.runtime.streamrecord.StreamRecord; import org.apache.flink.streaming.runtime.tasks.StreamTask; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/OneInputStreamOperatorTestHarness.java b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/OneInputStreamOperatorTestHarness.java index b83feca9df9105e600228640e668f43d9a6e29d8..01f95bc5ccc0ca214fd9256ec0320a8b2c810f4b 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/OneInputStreamOperatorTestHarness.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/OneInputStreamOperatorTestHarness.java @@ -30,8 +30,8 @@ import org.apache.flink.runtime.operators.testutils.MockInputSplitProvider; import org.apache.flink.streaming.api.graph.StreamConfig; import org.apache.flink.streaming.api.operators.OneInputStreamOperator; import org.apache.flink.streaming.api.operators.Output; -import org.apache.flink.streaming.api.state.StateBackend; -import org.apache.flink.streaming.api.state.memory.MemoryStateBackend; +import org.apache.flink.runtime.state.StateBackend; +import org.apache.flink.runtime.state.memory.MemoryStateBackend; import org.apache.flink.streaming.api.watermark.Watermark; import org.apache.flink.streaming.runtime.streamrecord.StreamRecord; import org.apache.flink.streaming.runtime.tasks.StreamTask; diff --git a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/TwoInputStreamOperatorTestHarness.java b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/TwoInputStreamOperatorTestHarness.java index 9b33c6a47512b288d3a9cc8047ffe435681fddb3..c586db36e36426a3bd6fc61994a3cb4da0416d78 100644 --- a/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/TwoInputStreamOperatorTestHarness.java +++ b/flink-staging/flink-streaming/flink-streaming-core/src/test/java/org/apache/flink/streaming/util/TwoInputStreamOperatorTestHarness.java @@ -28,8 +28,8 @@ import org.apache.flink.runtime.operators.testutils.MockInputSplitProvider; import org.apache.flink.streaming.api.graph.StreamConfig; import org.apache.flink.streaming.api.operators.Output; import org.apache.flink.streaming.api.operators.TwoInputStreamOperator; -import org.apache.flink.streaming.api.state.StateBackend; -import org.apache.flink.streaming.api.state.memory.MemoryStateBackend; +import org.apache.flink.runtime.state.StateBackend; +import org.apache.flink.runtime.state.memory.MemoryStateBackend; import org.apache.flink.streaming.api.watermark.Watermark; import org.apache.flink.streaming.runtime.streamrecord.StreamRecord; import org.apache.flink.streaming.runtime.tasks.StreamTask; diff --git a/flink-staging/flink-streaming/flink-streaming-scala/src/main/scala/org/apache/flink/streaming/api/scala/StreamExecutionEnvironment.scala b/flink-staging/flink-streaming/flink-streaming-scala/src/main/scala/org/apache/flink/streaming/api/scala/StreamExecutionEnvironment.scala index 29bf9389cc2d46c79430812f5922da0919ef66b5..e9536960acf9c56e31a7715026a2a14cd52fc17b 100644 --- a/flink-staging/flink-streaming/flink-streaming-scala/src/main/scala/org/apache/flink/streaming/api/scala/StreamExecutionEnvironment.scala +++ b/flink-staging/flink-streaming/flink-streaming-scala/src/main/scala/org/apache/flink/streaming/api/scala/StreamExecutionEnvironment.scala @@ -26,7 +26,7 @@ import org.apache.flink.api.common.io.{FileInputFormat, InputFormat} import org.apache.flink.api.common.typeinfo.TypeInformation import org.apache.flink.api.java.typeutils.runtime.kryo.KryoSerializer import org.apache.flink.api.scala.ClosureCleaner -import org.apache.flink.streaming.api.state.StateBackend +import org.apache.flink.runtime.state.StateBackend import org.apache.flink.streaming.api.{TimeCharacteristic, CheckpointingMode} import org.apache.flink.streaming.api.environment.{StreamExecutionEnvironment => JavaEnv} import org.apache.flink.streaming.api.functions.source.FileMonitoringFunction.WatchType diff --git a/flink-tests/src/test/java/org/apache/flink/test/classloading/ClassLoaderITCase.java b/flink-tests/src/test/java/org/apache/flink/test/classloading/ClassLoaderITCase.java index 989db147dd018c82d4bd7c1640abd65bf12aeefe..3b5295ff7b8701d0871c975cc4df8f0439c7ef5c 100644 --- a/flink-tests/src/test/java/org/apache/flink/test/classloading/ClassLoaderITCase.java +++ b/flink-tests/src/test/java/org/apache/flink/test/classloading/ClassLoaderITCase.java @@ -23,7 +23,7 @@ import java.io.File; import org.apache.flink.client.program.PackagedProgram; import org.apache.flink.configuration.ConfigConstants; import org.apache.flink.configuration.Configuration; -import org.apache.flink.streaming.api.state.filesystem.FsStateBackendFactory; +import org.apache.flink.runtime.state.filesystem.FsStateBackendFactory; import org.apache.flink.test.testdata.KMeansData; import org.apache.flink.test.util.ForkableFlinkMiniCluster; import org.junit.Rule; diff --git a/flink-tests/src/test/java/org/apache/flink/test/recovery/ChaosMonkeyITCase.java b/flink-tests/src/test/java/org/apache/flink/test/recovery/ChaosMonkeyITCase.java index 2cdf83c20e119c5efd0523140298e3e2ff66cf9f..f15644e69c38196f2564efb1f061b819f5e9a10b 100644 --- a/flink-tests/src/test/java/org/apache/flink/test/recovery/ChaosMonkeyITCase.java +++ b/flink-tests/src/test/java/org/apache/flink/test/recovery/ChaosMonkeyITCase.java @@ -32,6 +32,7 @@ import org.apache.flink.runtime.jobgraph.JobStatus; import org.apache.flink.runtime.leaderelection.TestingListener; import org.apache.flink.runtime.leaderretrieval.LeaderRetrievalService; import org.apache.flink.runtime.messages.JobManagerMessages; +import org.apache.flink.runtime.state.filesystem.FsStateBackendFactory; import org.apache.flink.runtime.testutils.CommonTestUtils; import org.apache.flink.runtime.testutils.JobManagerActorTestUtils; import org.apache.flink.runtime.testutils.JobManagerProcess; @@ -542,7 +543,7 @@ public class ChaosMonkeyITCase { LOG.info("Checking file system backend state..."); - File fsCheckpoints = new File(config.getString(ConfigConstants.STATE_BACKEND_FS_DIR, "")); + File fsCheckpoints = new File(config.getString(FsStateBackendFactory.CHECKPOINT_DIRECTORY_URI_CONF_KEY, "")); LOG.info("Checking " + fsCheckpoints); diff --git a/flink-tests/src/test/java/org/apache/flink/test/recovery/JobManagerCheckpointRecoveryITCase.java b/flink-tests/src/test/java/org/apache/flink/test/recovery/JobManagerCheckpointRecoveryITCase.java index 54ddf7e300b1c04bafaa70730a28248c929fa44a..e59006743251b194b27050fda1b75cb62cf79293 100644 --- a/flink-tests/src/test/java/org/apache/flink/test/recovery/JobManagerCheckpointRecoveryITCase.java +++ b/flink-tests/src/test/java/org/apache/flink/test/recovery/JobManagerCheckpointRecoveryITCase.java @@ -44,6 +44,7 @@ import org.apache.flink.streaming.api.checkpoint.Checkpointed; import org.apache.flink.streaming.api.environment.StreamExecutionEnvironment; import org.apache.flink.streaming.api.functions.sink.SinkFunction; import org.apache.flink.streaming.api.functions.source.RichParallelSourceFunction; +import org.apache.flink.util.TestLogger; import org.junit.AfterClass; import org.junit.Before; import org.junit.Test; @@ -68,7 +69,7 @@ import static org.apache.flink.runtime.messages.JobManagerMessages.SubmitJob; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertNotEquals; -public class JobManagerCheckpointRecoveryITCase { +public class JobManagerCheckpointRecoveryITCase extends TestLogger { private final static ZooKeeperTestEnvironment ZooKeeper = new ZooKeeperTestEnvironment(1); @@ -144,7 +145,7 @@ public class JobManagerCheckpointRecoveryITCase { JobGraph jobGraph = env.getStreamGraph().getJobGraph(); Configuration config = ZooKeeperTestUtils.createZooKeeperRecoveryModeConfig(ZooKeeper - .getConnectString(), FileStateBackendBasePath.getPath()); + .getConnectString(), FileStateBackendBasePath.getAbsoluteFile().toURI().toString()); config.setInteger(ConfigConstants.TASK_MANAGER_NUM_TASK_SLOTS, Parallelism); ActorSystem testSystem = null; diff --git a/flink-tests/src/test/java/org/apache/flink/test/recovery/TaskManagerProcessFailureStreamingRecoveryITCase.java b/flink-tests/src/test/java/org/apache/flink/test/recovery/TaskManagerProcessFailureStreamingRecoveryITCase.java index aa634f0e321b063600bd5d69c227a41da72290e0..63aa96788554f1ee64c3a488eb43e2a9e29ecbaf 100644 --- a/flink-tests/src/test/java/org/apache/flink/test/recovery/TaskManagerProcessFailureStreamingRecoveryITCase.java +++ b/flink-tests/src/test/java/org/apache/flink/test/recovery/TaskManagerProcessFailureStreamingRecoveryITCase.java @@ -33,7 +33,7 @@ import org.apache.flink.streaming.api.datastream.DataStream; import org.apache.flink.streaming.api.environment.StreamExecutionEnvironment; import org.apache.flink.streaming.api.functions.sink.RichSinkFunction; import org.apache.flink.streaming.api.functions.source.RichParallelSourceFunction; -import org.apache.flink.streaming.api.state.filesystem.FsStateBackend; +import org.apache.flink.runtime.state.filesystem.FsStateBackend; import org.junit.Assert; diff --git a/flink-tests/src/test/java/org/apache/flink/test/runtime/leaderelection/ZooKeeperLeaderElectionITCase.java b/flink-tests/src/test/java/org/apache/flink/test/runtime/leaderelection/ZooKeeperLeaderElectionITCase.java index 5c7a93277622f83d7aacea0d418b8669751850ac..5840a98dfe812aaf6d961f2c7667c3d41fb66c52 100644 --- a/flink-tests/src/test/java/org/apache/flink/test/runtime/leaderelection/ZooKeeperLeaderElectionITCase.java +++ b/flink-tests/src/test/java/org/apache/flink/test/runtime/leaderelection/ZooKeeperLeaderElectionITCase.java @@ -95,7 +95,7 @@ public class ZooKeeperLeaderElectionITCase extends TestLogger { configuration.setInteger(ConfigConstants.LOCAL_NUMBER_JOB_MANAGER, numJMs); configuration.setInteger(ConfigConstants.LOCAL_NUMBER_TASK_MANAGER, numTMs); configuration.setString(ConfigConstants.STATE_BACKEND, "filesystem"); - configuration.setString(ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, tempDirectory.getPath()); + configuration.setString(ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, tempDirectory.getAbsoluteFile().toURI().toString()); ForkableFlinkMiniCluster cluster = new ForkableFlinkMiniCluster(configuration); @@ -144,7 +144,7 @@ public class ZooKeeperLeaderElectionITCase extends TestLogger { configuration.setInteger(ConfigConstants.LOCAL_NUMBER_TASK_MANAGER, numTMs); configuration.setInteger(ConfigConstants.TASK_MANAGER_NUM_TASK_SLOTS, numSlotsPerTM); configuration.setString(ConfigConstants.STATE_BACKEND, "filesystem"); - configuration.setString(ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, tempDirectory.getPath()); + configuration.setString(ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH, tempDirectory.getAbsoluteFile().toURI().toString()); // we "effectively" disable the automatic RecoverAllJobs message and sent it manually to make // sure that all TMs have registered to the JM prior to issueing the RecoverAllJobs message diff --git a/flink-tests/src/test/resources/log4j-test.properties b/flink-tests/src/test/resources/log4j-test.properties index 3ba6d1d11c15e5d3cdd3f362bc570882d9a2911b..62807b321d80e49248d8927f5df9a192e4abca0c 100644 --- a/flink-tests/src/test/resources/log4j-test.properties +++ b/flink-tests/src/test/resources/log4j-test.properties @@ -19,7 +19,6 @@ # Set root logger level to OFF to not flood build logs # set manually to INFO for debugging purposes log4j.rootLogger=INFO, testlogger -log4j.logger.org.apache.flink.runtime.client.JobClientActor=DEBUG # A1 is set to be a ConsoleAppender. log4j.appender.testlogger=org.apache.log4j.ConsoleAppender diff --git a/flink-yarn-tests/src/main/java/org/apache/flink/yarn/YARNHighAvailabilityITCase.java b/flink-yarn-tests/src/main/java/org/apache/flink/yarn/YARNHighAvailabilityITCase.java index a05621a985d5e89c61ef9bc1a269c900907c249a..ffb43f8ef0f83da2a0f17b00490180a6e2164c09 100644 --- a/flink-yarn-tests/src/main/java/org/apache/flink/yarn/YARNHighAvailabilityITCase.java +++ b/flink-yarn-tests/src/main/java/org/apache/flink/yarn/YARNHighAvailabilityITCase.java @@ -30,6 +30,7 @@ import org.apache.flink.runtime.instance.ActorGateway; import org.apache.flink.runtime.instance.AkkaActorGateway; import org.apache.flink.runtime.leaderretrieval.LeaderRetrievalService; import org.apache.flink.runtime.messages.Messages; +import org.apache.flink.runtime.state.filesystem.FsStateBackendFactory; import org.apache.flink.runtime.testingUtils.TestingJobManagerMessages; import org.apache.flink.runtime.util.LeaderRetrievalUtils; import org.apache.flink.runtime.yarn.AbstractFlinkYarnCluster; @@ -114,7 +115,7 @@ public class YARNHighAvailabilityITCase extends YarnTestBase { flinkYarnClient.setDynamicPropertiesEncoded("recovery.mode=zookeeper@@ha.zookeeper.quorum=" + zkServer.getConnectString() + "@@yarn.application-attempts=" + numberApplicationAttempts + "@@" + ConfigConstants.STATE_BACKEND + "=FILESYSTEM" + - "@@" + ConfigConstants.STATE_BACKEND_FS_DIR + "=" + fsStateHandlePath + "/checkpoints" + + "@@" + FsStateBackendFactory.CHECKPOINT_DIRECTORY_URI_CONF_KEY + "=" + fsStateHandlePath + "/checkpoints" + "@@" + ConfigConstants.STATE_BACKEND_FS_RECOVERY_PATH + "=" + fsStateHandlePath + "/recovery"); flinkYarnClient.setConfigurationFilePath(new Path(confDirPath + File.separator + "flink-conf.yaml")); diff --git a/flink-yarn/src/main/scala/org/apache/flink/yarn/ApplicationMasterBase.scala b/flink-yarn/src/main/scala/org/apache/flink/yarn/ApplicationMasterBase.scala index 84198ec55bbc0660c474f616468c867d236433e0..f3892dd2025496652565c26cd4ba566c596205df 100644 --- a/flink-yarn/src/main/scala/org/apache/flink/yarn/ApplicationMasterBase.scala +++ b/flink-yarn/src/main/scala/org/apache/flink/yarn/ApplicationMasterBase.scala @@ -123,7 +123,7 @@ abstract class ApplicationMasterBase { config.setInteger(ConfigConstants.JOB_MANAGER_WEB_PORT_KEY, 0); // set port to 0. } - val (actorSystem, jmActor, archivActor, webMonitor) = + val (actorSystem, jmActor, archiveActor, webMonitor) = JobManager.startActorSystemAndJobManagerActors( config, JobManagerMode.CLUSTER, diff --git a/pom.xml b/pom.xml index 47c47c55891257e12655023cc9cf37432c17f76f..efe2101d2934517f8d37a1ba495f148029d4706b 100644 --- a/pom.xml +++ b/pom.xml @@ -735,7 +735,7 @@ under the License. flink-runtime-web/web-dashboard/assets/fonts/fontawesome* - flink-runtime-web/src/main/resources/web/** + flink-runtime-web/web-dashboard/web/** flink-runtime-web/web-dashboard/node_modules/**