/// /// module Kubernetes { var log = Logger.get("kubernetes-pod-logs"); _module.service("PodLogReplacements", () => { return []; }); _module.run((PodLogReplacements) => { var log = Logger.get("pod-log-replacers"); // Add ANSI escape character replacer // adapted from https://github.com/mmalecki/ansispan var colors = { '30': 'black', '31': 'red', '32': 'green', '33': 'yellow', '34': 'blue', '35': 'purple', '36': 'cyan', '37': 'white' } PodLogReplacements.push((msg) => { if (!msg) { return msg; } var end = ""; _.forOwn(colors, (color, code) => { var start = ``; msg = msg.replace(new RegExp('\033\\[' + code + 'm', 'g'), start) msg = msg.replace(new RegExp('\033\\[0;' + code + 'm', 'g'), start); }); msg = msg.replace(/\033\[1m/g, '').replace(/\033\[22m/g, ''); msg = msg.replace(/\033\[3m/g, '').replace(/\033\[23m/g, ''); msg = msg.replace(/\033\[m/g, end); msg = msg.replace(/\033\[0m/g, end); msg = msg.replace(/\033\[39m/g, end); msg = msg.replace(/\033\[2m/g, ''); msg = msg.replace(/\033\[0;39m/g, end); log.debug("Running replacement on message: ", msg); return msg; }); }); _module.controller("Kubernetes.PodLogLinkController", ($scope, TerminalService, $templateCache) => { $scope.openLogs = (entity) => { log.debug("Open logs: ", entity); TerminalService.newTerminal(entity.metadata.selfLink, entity.metadata.name, $templateCache.get(UrlHelpers.join(templatePath, 'logShell.html'))); } }); _module.directive('podLogDisplay', (userDetails, PodLogReplacements) => { return { restrict: 'E', template: `

Please wait, fetching logs...

View previous container logs?

`, link: (scope:any, element, attr) => { var link = scope.$eval('podLink'); var name = scope.$eval('containerName'); if (!link) { return; } scope.fetched = false; scope.previous = false; scope.messages = []; link = UrlHelpers.join(masterApiUrl(), link, 'log'); link = KubernetesAPI.wsUrl(link); link.search({ follow: true, tailLines: 1000, access_token: userDetails.token }); var messages = []; var pullMessages = _.debounce(() => { scope.messages = scope.messages.concat(_.remove(messages, () => true).map((msg) => { PodLogReplacements.forEach((replFunc:any) => { if (angular.isFunction(replFunc)) { msg = replFunc(msg); } }); return msg; })); if (!scope.fetched) { scope.fetched = true; } Core.$apply(scope); }, 1000); function initSocket(link) { scope.fetched = false; messages.length = 0; scope.messages.length = 0; var ws = new WebSocket(link.toString(), 'base64.binary.k8s.io'); ws.onmessage = (event) => { try { var message = window.atob(event.data); messages.push(message); pullMessages(); } catch (err) { // we'll just ignore these //log.debug("Failed to decode message: ", event.data, " error: ", err); } } return ws; } var ws = initSocket(link); scope.$watch('previous', (value, old) => { if (value !== old) { if (link.hasSearch('previous')) { link.removeSearch('previous').addSearch('previous', scope.previous); } else { link.addSearch('previous', scope.previous); } ws.close(); ws = initSocket(link); } }); element.on('$destroy', () => { if (ws) { try { ws.close(); } catch (err) { // nothing to do } delete ws; } }); } } }); _module.directive('podLogWindow', ($compile, TerminalService) => { return { restrict: 'A', scope: false, link: (scope:any, element, attr) => { addWindowActions(scope, element, TerminalService); scope.atBottom = true; scope.$watch('atBottom', (val) => { console.log("atBottom: ", val); }); } }; }); }