UNPKG

ansi-to-html

Version:

Convert ansi escaped text streams to html.

493 lines (356 loc) 18.2 kB
'use strict'; const {expect} = require('chai'); const Filter = require('../lib/ansi_to_html.js'); function test(text, result, done, opts) { if (!opts) { opts = {}; } const f = new Filter(opts); function filtered(memo, t) { return memo + f.toHtml(t); } text = typeof text.reduce === 'function' ? text : [text]; expect(text.reduce(filtered, '')).to.equal(result); return done(); } describe('ansi to html', function () { describe('constructed with no options', function () { it('doesn\'t modify the input string', function (done) { const text = 'some text'; const result = 'some text'; return test(text, result, done); }); it('returns plain text when given plain text with LF', function (done) { const text = 'test\ntest\n'; const result = 'test\ntest\n'; return test(text, result, done); }); it('returns plain text when given plain text with multiple LF', function (done) { const text = 'test\n\n\ntest\n'; const result = 'test\n\n\ntest\n'; return test(text, result, done); }); it('returns plain text when given plain text with CR', function (done) { const text = 'testCRLF\rtest'; const result = 'testCRLF\rtest'; return test(text, result, done); }); it('returns plain text when given plain text with multiple CR', function (done) { const text = 'testCRLF\r\r\rtest'; const result = 'testCRLF\r\r\rtest'; return test(text, result, done); }); it('returns plain text when given plain text with CR & LF', function (done) { const text = 'testCRLF\r\ntest'; const result = 'testCRLF\r\ntest'; return test(text, result, done); }); it('returns plain text when given plain text with multiple CR & LF', function (done) { const text = 'testCRLF\r\n\r\ntest'; const result = 'testCRLF\r\n\r\ntest'; return test(text, result, done); }); it('renders foreground colors', function (done) { const text = 'colors: \x1b[30mblack\x1b[37mwhite'; const result = 'colors: <span style="color:#000">black<span style="color:#AAA">white</span></span>'; return test(text, result, done); }); it('renders light foreground colors', function (done) { const text = 'colors: \x1b[90mblack\x1b[97mwhite'; const result = 'colors: <span style="color:#555">black<span style="color:#FFF">white</span></span>'; return test(text, result, done); }); it('renders background colors', function (done) { const text = 'colors: \x1b[40mblack\x1b[47mwhite'; const result = 'colors: <span style="background-color:#000">black<span style="background-color:#AAA">white</span></span>'; return test(text, result, done); }); it('renders light background colors', function (done) { const text = 'colors: \x1b[100mblack\x1b[107mwhite'; const result = 'colors: <span style="background-color:#555">black<span style="background-color:#FFF">white</span></span>'; return test(text, result, done); }); it('renders strikethrough', function (done) { const text = 'strike: \x1b[9mthat'; const result = 'strike: <strike>that</strike>'; return test(text, result, done); }); it('renders blink', function (done) { const text = 'blink: \x1b[5mwhat'; const result = 'blink: <blink>what</blink>'; return test(text, result, done); }); it('renders underline', function (done) { const text = 'underline: \x1b[4mstuff'; const result = 'underline: <u>stuff</u>'; return test(text, result, done); }); it('renders bold', function (done) { const text = 'bold: \x1b[1mstuff'; const result = 'bold: <b>stuff</b>'; return test(text, result, done); }); it('renders italic', function (done) { const text = 'italic: \x1b[3mstuff'; const result = 'italic: <i>stuff</i>'; return test(text, result, done); }); it('handles resets', function (done) { const text = '\x1b[1mthis is bold\x1b[0m, but this isn\'t'; const result = '<b>this is bold</b>, but this isn\'t'; return test(text, result, done); }); it('handles multiple resets', function (done) { const text = 'normal, \x1b[1mbold, \x1b[4munderline, \x1b[31mred\x1b[0m, normal'; const result = 'normal, <b>bold, <u>underline, <span style="color:' + '#A00">red</span></u></b>, normal'; return test(text, result, done); }); it('handles resets with implicit 0', function (done) { const text = '\x1b[1mthis is bold\x1b[m, but this isn\'t'; const result = '<b>this is bold</b>, but this isn\'t'; return test(text, result, done); }); it('renders multi-attribute sequences', function (done) { const text = 'normal, \x1b[1;4;31mbold, underline, and red\x1b[0m, normal'; const result = 'normal, <b><u><span style="color:#A00">bold, underline,' + ' and red</span></u></b>, normal'; return test(text, result, done); }); it('renders multi-attribute sequences with a semi-colon', function (done) { const text = 'normal, \x1b[1;4;31;mbold, underline, and red\x1b[0m, normal'; const result = 'normal, <b><u><span style="color:#A00">bold, underline, and red</span></u></b>, normal'; return test(text, result, done); }); it('eats malformed sequences', function (done) { const text = '\x1b[25oops forgot the \'m\''; const result = 'oops forgot the \'m\''; return test(text, result, done); }); it('renders xterm 256 foreground sequences', function (done) { const text = '\x1b[38;5;196mhello'; const result = '<span style="color:#ff0000">hello</span>'; return test(text, result, done); }); it('renders xterm 256 background sequences', function (done) { const text = '\x1b[48;5;196mhello'; const result = '<span style="background-color:#ff0000">hello</span>'; return test(text, result, done); }); it('renders foreground rgb sequences', function (done) { const text = '\x1b[38;2;210;60;114mhello'; const result = '<span style="color:#d23c72">hello</span>'; return test(text, result, done); }); it('renders background rgb sequences', function (done) { const text = '\x1b[48;2;155;42;45mhello'; const result = '<span style="background-color:#9b2a2d">hello</span>'; return test(text, result, done); }); it('handles resetting to default foreground color', function (done) { const text = '\x1b[30mblack\x1b[39mdefault'; const result = '<span style="color:#000">black<span style="color:#FFF">default</span></span>'; return test(text, result, done); }); it('handles resetting to default background color', function (done) { const text = '\x1b[100mblack\x1b[49mdefault'; const result = '<span style="background-color:#555">black<span style="background-color:#000">default</span></span>'; return test(text, result, done); }); it('is able to disable underline', function (done) { const text = 'underline: \x1b[4mstuff\x1b[24mthings'; const result = 'underline: <u>stuff</u>things'; return test(text, result, done); }); it('is able to skip disabling underline', function (done) { const text = 'not underline: stuff\x1b[24mthings'; const result = 'not underline: stuffthings'; return test(text, result, done); }); it('renders two escape sequences in sequence', function (done) { const text = 'months remaining\x1b[1;31mtimes\x1b[m\x1b[1;32mmultiplied by\x1b[m $10'; const result = 'months remaining<b><span style="color:#A00">times</span></b><b><span style="color:#0A0">multiplied by</span></b> $10'; return test(text, result, done); }); it('drops EL code with no parameter', function (done) { const text = '\x1b[Khello'; const result = 'hello'; return test(text, result, done); }); it('drops EL code with 0 parameter', function (done) { const text = '\x1b[0Khello'; const result = 'hello'; return test(text, result, done); }); it('drops EL code with 0 parameter after new line character', function (done) { const text = 'HELLO\n\x1b[0K\u001b[33;1mWORLD\u001b[0m\n'; const result = 'HELLO\n<span style="color:#A50"><b>WORLD</b></span>\n'; return test(text, result, done); }); it('drops EL code with 1 parameter', function (done) { const text = '\x1b[1Khello'; const result = 'hello'; return test(text, result, done); }); it('drops EL code with 2 parameter', function (done) { const text = '\x1b[2Khello'; const result = 'hello'; return test(text, result, done); }); it('drops ED code with 0 parameter', function (done) { const text = '\x1b[Jhello'; const result = 'hello'; return test(text, result, done); }); it('drops ED code with 1 parameter', function (done) { const text = '\x1b[1Jhello'; const result = 'hello'; return test(text, result, done); }); it('drops HVP code with 0 parameter', function (done) { const text = '\x1b[;fhello'; const result = 'hello'; return test(text, result, done); }); it('drops HVP code with 1 parameter', function (done) { const text = '\x1b[123;fhello'; const result = 'hello'; return test(text, result, done); }); it('drops HVP code with 2 parameter', function (done) { const text = '\x1b[123;456fhello'; const result = 'hello'; return test(text, result, done); }); it('drops setusg0 sequence', function (done) { const text = '\x1b[(Bhello'; const result = 'hello'; return test(text, result, done); }); it('renders un-italic code appropriately', function (done) { const text = '\x1b[3mHello\x1b[23m World'; const result = '<i>Hello</i> World'; return test(text, result, done); }); it('skips rendering un-italic code appropriately', function (done) { const text = 'Hello\x1b[23m World'; const result = 'Hello World'; return test(text, result, done); }); it('renders overline', function (done) { const text = '\x1b[53mHello World'; const result = '<span style="text-decoration:overline">Hello World</span>'; return test(text, result, done); }); it('renders normal text', function (done) { const text = '\x1b[22mnormal text'; const result = '<span style="font-weight:normal;text-decoration:none;font-style:normal">normal text</span>'; return test(text, result, done); }); it('renders text following carriage return (CR, mac style line break)', function (done) { const text = 'ANSI Hello\rWorld'; const result = 'ANSI Hello\rWorld'; return test(text, result, done); }); }); describe('with escapeXML option enabled', function () { it('escapes XML entities', function (done) { const text = 'normal, \x1b[1;4;31;mbold, <underline>, and red\x1b[0m, normal'; const result = 'normal, <b><u><span style="color:#A00">bold, &lt;underline&gt;, and red</span></u></b>, normal'; return test(text, result, done, {escapeXML: true}); }); }); describe('with newline option enabled', function () { it('renders line breaks', function (done) { const text = 'test\ntest\n'; const result = 'test<br/>test<br/>'; return test(text, result, done, {newline: true}); }); it('renders multiple line breaks', function (done) { const text = 'test\n\ntest\n'; const result = 'test<br/><br/>test<br/>'; return test(text, result, done, {newline: true}); }); it('renders mac styled line breaks (CR)', function (done) { const text = 'test\rtest\r'; const result = 'test<br/>test<br/>'; return test(text, result, done, {newline: true}); }); it('renders multiple mac styled line breaks (CR)', function (done) { const text = 'test\r\rtest\r'; const result = 'test<br/><br/>test<br/>'; return test(text, result, done, {newline: true}); }); it('renders windows styled line breaks (CR+LF)', function (done) { const text = 'testCRLF\r\ntestLF'; const result = 'testCRLF<br/>testLF'; return test(text, result, done, {newline: true}); }); it('renders windows styled line breaks (multi CR+LF)', function (done) { const text = 'testCRLF\r\n\r\ntestLF'; const result = 'testCRLF<br/><br/>testLF'; return test(text, result, done, {newline: true}); }); }); describe('with stream option enabled', function () { it('persists styles between toHtml() invocations', function (done) { const text = ['\x1b[31mred', 'also red']; const result = '<span style="color:#A00">red</span><span style="color:#A00">also red</span>'; return test(text, result, done, {stream: true}); }); it('persists styles between more than two toHtml() invocations', function (done) { const text = ['\x1b[31mred', 'also red', 'and red']; const result = '<span style="color:#A00">red</span><span style="color:#A00">also red</span><span style="color:#A00">and red</span>'; return test(text, result, done, {stream: true}); }); it('does not persist styles beyond their usefulness', function (done) { const text = ['\x1b[31mred', 'also red', '\x1b[30mblack', 'and black']; const result = '<span style="color:#A00">red</span><span style="color:#A00">also red</span><span style="color:#A00"><span style="color:#000">black</span></span><span style="color:#000">and black</span>'; return test(text, result, done, {stream: true}); }); it('removes one state when encountering a reset', function (done) { const text = ['\x1b[1mthis is bold\x1b[0m, but this isn\'t', ' nor is this']; const result = '<b>this is bold</b>, but this isn\'t nor is this'; return test(text, result, done, {stream: true}); }); it('removes multiple state when encountering a reset', function (done) { const text = ['\x1b[1mthis \x1b[9mis bold\x1b[0m, but this isn\'t', ' nor is this']; const result = '<b>this <strike>is bold</strike></b>, but this isn\'t nor is this'; return test(text, result, done, {stream: true}); }); }); describe('with custom colors enabled', function () { it('renders basic colors', function (done) { const text = ['\x1b[31mblue', 'not blue']; const result = '<span style="color:#00A">blue</span>not blue'; return test(text, result, done, {colors: {1: '#00A'}}); }); it('renders basic colors with streaming', function (done) { const text = ['\x1b[31mblue', 'also blue']; const result = '<span style="color:#00A">blue</span><span style="color:#00A">also blue</span>'; return test(text, result, done, {stream: true, colors: {1: '#00A'}}); }); it('renders custom colors and default colors', function (done) { const text = ['\x1b[31mblue', 'not blue', '\x1b[94mlight blue', 'not colored']; const result = '<span style="color:#00A">blue</span>not blue<span style="color:#55F">light blue</span>not colored'; return test(text, result, done, {colors: {1: '#00A'}}); }); it('renders custom colors and default colors together', function (done) { const text = ['\x1b[31mblue', 'not blue', '\x1b[94mlight blue', 'not colored']; const result = '<span style="color:#00A">blue</span>not blue<span style="color:#55F">light blue</span>not colored'; return test(text, result, done, {colors: {1: '#00A'}}); }); it('renders custom 8/ 16 colors', function (done) { // code - 90 + 8 = color // so 94 - 90 + 8 = 12 const text = ['\x1b[94mlighter blue']; const result = '<span style="color:#33F">lighter blue</span>'; return test(text, result, done, {colors: {12: '#33F'}}); }); it('renders custom 256 colors', function (done) { // code - 90 + 8 = color // so 94 - 90 + 8 = 12 const text = ['\x1b[38;5;125mdark red', 'then \x1b[38;5;126msome other color']; const result = '<span style="color:#af005f">dark red</span>then <span style="color:#af225f">some other color</span>'; return test(text, result, done, {colors: {126: '#af225f'}}); }); }); });