PHP and Javascript implementations of a simple markdown parser
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

testjs.html 20KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640
  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="utf-8">
  5. <title>Markdown Unit Tests</title>
  6. <link rel="icon" href="data:;base64,iVBORw0KGgo=">
  7. <style type="text/css">
  8. :root {
  9. font-family: sans-serif;
  10. }
  11. .testclass {
  12. border: 1px solid black;
  13. padding: 0.5em 1em;
  14. margin-bottom: 1em;
  15. max-width: 50em;
  16. }
  17. .testclassname {
  18. font-weight: bold;
  19. font-size: 1.25rem;
  20. padding-bottom: 0.25em;
  21. }
  22. .testcase {
  23. padding: 0.2em 0;
  24. margin-left: 2em;
  25. }
  26. .testcase {
  27. border-top: 1px solid #888;
  28. }
  29. .testcasename {
  30. font-size: 115%;
  31. font-weight: bold;
  32. }
  33. .testcasestatus {
  34. font-weight: bold;
  35. }
  36. .result-untested {
  37. color: #888;
  38. }
  39. .result-testing {
  40. color: black;
  41. }
  42. .result-passed {
  43. color: #090;
  44. }
  45. .result-failed {
  46. color: #a00;
  47. }
  48. .result-errored {
  49. color: #a80;
  50. }
  51. </style>
  52. <script src="js/markdown.js"></script>
  53. <!-- Testing infrastructure -->
  54. <script>
  55. /**
  56. * @param {String} text
  57. * @returns {String}
  58. */
  59. function escapeHTML(text) {
  60. return text.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\n/g, '<br/>\n');
  61. }
  62. class ResultType {
  63. static untested = new ResultType('untested');
  64. static testing = new ResultType('testing');
  65. static passed = new ResultType('passed');
  66. static failed = new ResultType('failed');
  67. static errored = new ResultType('errored');
  68. #name;
  69. constructor(name) {
  70. this.#name = name;
  71. }
  72. toString() {
  73. return `${this.constructor.name}.${this.#name}`;
  74. }
  75. }
  76. class FailureError extends Error {
  77. constructor(message=null) {
  78. super(message);
  79. }
  80. }
  81. class BaseTest {
  82. setUp() {}
  83. tearDown() {}
  84. /** @var {TestCaseRunner|null} */
  85. currentRunner = null;
  86. fail(failMessage=null) {
  87. throw new FailureError(failMessage || 'failed');
  88. }
  89. assertTrue(test, failMessage=null) {
  90. if (!test) this.fail(failMessage || `expected true, got ${test}`);
  91. }
  92. assertFalse(test, failMessage=null) {
  93. if (test) this.fail(failMessage || `expected false, got ${test}`);
  94. }
  95. assertEqual(a, b, failMessage=null) {
  96. if (a == b) return;
  97. const aVal = `${a}`;
  98. const bVal = `${b}`;
  99. if (aVal.length > 20 || bVal.length > 20) {
  100. this.fail(failMessage || `equality failed:\n${aVal}\n!=\n${bVal}`);
  101. } else {
  102. this.fail(failMessage || `equality failed: ${aVal} != ${bVal}`);
  103. }
  104. }
  105. expectError(e=true) {
  106. if (this.currentRunner) this.currentRunner.expectedError = e;
  107. }
  108. /**
  109. * @param {number} maxTime maximum time in seconds
  110. * @param {function} timedCode code to run and time
  111. */
  112. profile(maxTime, timedCode) {
  113. const startTime = performance.now();
  114. const result = timedCode();
  115. const endTime = performance.now();
  116. const seconds = (endTime - startTime) / 1000.0;
  117. if (seconds > maxTime) {
  118. this.fail(`Expected <= ${maxTime}s execution time, actual ${seconds}s.`);
  119. }
  120. return result;
  121. }
  122. }
  123. /**
  124. * Manages the running and results of a single test method on a test
  125. * class.
  126. */
  127. class TestCaseRunner {
  128. /** @var {number} */
  129. static #nextUniqueId = 1;
  130. /** @var {number} */
  131. uniqueId = 0;
  132. /** @var {BaseTest} */
  133. #objectUnderTest;
  134. /** @var {method} */
  135. #method;
  136. /** @var {ResultType} */
  137. result = ResultType.untested;
  138. /** @var {String|null} */
  139. message = null;
  140. expectedError = null;
  141. /** @var {String} */
  142. get className() { return this.#objectUnderTest.constructor.name; }
  143. /** @var {String} */
  144. get methodName() { return this.#method.name; }
  145. /**
  146. * @param {BaseTest} objectUnderTest
  147. * @param {method} method
  148. */
  149. constructor(objectUnderTest, method) {
  150. this.#objectUnderTest = objectUnderTest;
  151. this.#method = method;
  152. this.uniqueId = TestCaseRunner.#nextUniqueId++;
  153. }
  154. run() {
  155. try {
  156. this.expectedError = null;
  157. this.#objectUnderTest.currentRunner = this;
  158. this.#objectUnderTest.setUp();
  159. this.#method.bind(this.#objectUnderTest)();
  160. this.result = ResultType.passed;
  161. this.message = null;
  162. } catch (e) {
  163. if (e instanceof FailureError) {
  164. this.result = ResultType.failed;
  165. this.message = e.message;
  166. } else if (this.#isExpectedError(e)) {
  167. this.result = ResultType.passed;
  168. this.message = null;
  169. } else {
  170. this.result = ResultType.errored;
  171. this.message = e.message;
  172. if (e.stack !== undefined) {
  173. this.message += "\n" + e.stack;
  174. }
  175. }
  176. } finally {
  177. this.expectedError = null;
  178. try {
  179. this.#objectUnderTest.tearDown();
  180. this.#objectUnderTest.currentRunner = null;
  181. } catch (e0) {
  182. console.error(`Failed to run ${this.className}.tearDown() - ${e0.message}`);
  183. this.result = ResultType.errored;
  184. this.message = e0.message;
  185. }
  186. }
  187. }
  188. get #cssId() { return `testcase${this.uniqueId}`; }
  189. #isExpectedError(e) {
  190. if (this.expectedError === null) return false;
  191. if (this.expectedError === true) return true;
  192. // TODO: Have a way to specify details about what kind of error is expected. Maybe a prototype instance and/or a testing lambda.
  193. return false;
  194. }
  195. toHTML() {
  196. var html = `<div class="testcase" id="${this.#cssId}">`;
  197. html += `<div class="testcasename"><span class="testcasemethod">${this.methodName}</span></div>`;
  198. switch (this.result) {
  199. case ResultType.untested:
  200. html += '<div class="testcasestatus result-untested">Waiting to test</div>';
  201. break;
  202. case ResultType.testing:
  203. html += '<div class="testcasestatus result-tesitng">Testing...</div>';
  204. break;
  205. case ResultType.passed:
  206. html += '<div class="testcasestatus result-passed">Passed</div>';
  207. break;
  208. case ResultType.failed:
  209. html += '<div class="testcasestatus result-failed">Failed</div>';
  210. html += `<div class="testcasemessage">${escapeHTML(this.message)}</div>`;
  211. break;
  212. case ResultType.errored:
  213. html += '<div class="testcasestatus result-errored">Errored</div>';
  214. html += `<div class="testcasemessage">${escapeHTML(this.message)}</div>`;
  215. break;
  216. }
  217. html += `</div>`;
  218. return html;
  219. }
  220. /**
  221. * Updates the HTML node in-place with the current status, or
  222. * adds it if it does not exist yet.
  223. */
  224. updateHTML() {
  225. let existing = document.getElementById(this.#cssId);
  226. if (existing) {
  227. existing.outerHTML = this.toHTML();
  228. } else {
  229. document.getElementById('results').innerHTML += this.toHTML();
  230. }
  231. }
  232. /**
  233. * @param {object} objectUnderTest
  234. * @returns {TestCaseRunner[]}
  235. */
  236. static findTestCases(objectUnderTest) {
  237. if (!(objectUnderTest instanceof BaseTest)) return [];
  238. var members = [];
  239. var obj = objectUnderTest;
  240. do {
  241. members.push(...Object.getOwnPropertyNames(obj));
  242. } while (obj = Object.getPrototypeOf(obj));
  243. return members.sort().filter((e, i, arr) => {
  244. if (e != arr[i + 1] && typeof objectUnderTest[e] == 'function' && e.startsWith('test')) return true;
  245. }).map((name) => {
  246. return new TestCaseRunner(objectUnderTest, objectUnderTest[name]);
  247. });
  248. }
  249. }
  250. class TestClassRunner {
  251. static #nextUniqueId = 1;
  252. #uniqueId = 0;
  253. #theClass;
  254. #instance;
  255. #testCases;
  256. get testCases() { return this.#testCases; }
  257. constructor(theClass) {
  258. this.#theClass = theClass;
  259. this.#instance = new theClass();
  260. this.#testCases = TestCaseRunner.findTestCases(this.#instance);
  261. this.#uniqueId = TestClassRunner.#nextUniqueId++;
  262. }
  263. get #cssId() { return `testclass${this.#uniqueId}`; }
  264. toHTML() {
  265. var html = '';
  266. html += `<div class="testclass" id="${this.#cssId}">`;
  267. html += `<div class="testclassname">${this.#theClass.name}</div>`;
  268. for (const testCase of this.#testCases) {
  269. html += testCase.toHTML();
  270. }
  271. html += '</div>';
  272. return html;
  273. }
  274. updateHTML() {
  275. var existing = document.getElementById(this.#cssId);
  276. if (!existing) {
  277. document.getElementById('results').innerHTML += this.toHTML();
  278. }
  279. }
  280. static runAll(testClasses) {
  281. var tests = []; // tuples of TestClassRunner and TestCaseRunner
  282. for (const testClass of testClasses) {
  283. const classRunner = new TestClassRunner(testClass);
  284. classRunner.updateHTML();
  285. tests.push(...classRunner.testCases.map(function(test) { return [ classRunner, test ] }));
  286. }
  287. var testInterval = setInterval(function() {
  288. if (tests.length == 0) {
  289. clearInterval(testInterval);
  290. testInterval = null;
  291. return;
  292. }
  293. var classRunner;
  294. var testCase;
  295. [ classRunner, testCase ] = tests[0];
  296. if (testCase.result == ResultType.untested) {
  297. testCase.result = ResultType.testing;
  298. testCase.updateHTML();
  299. classRunner.updateHTML();
  300. } else if (testCase.result == ResultType.testing) {
  301. tests.splice(0, 1);
  302. testCase.run();
  303. testCase.updateHTML();
  304. classRunner.updateHTML();
  305. }
  306. }, 1);
  307. }
  308. }
  309. </script>
  310. <!-- Tests -->
  311. <script>
  312. function onLoad() {
  313. let testClasses = [
  314. TokenTests,
  315. InlineTests,
  316. BlockTests,
  317. ];
  318. TestClassRunner.runAll(testClasses);
  319. }
  320. document.addEventListener('DOMContentLoaded', onLoad);
  321. function normalizeWhitespace(str) {
  322. return str.replace(/\s+/g, ' ').replace(/(?:^\s+|\s+$)/g, '');
  323. }
  324. class TokenTests extends BaseTest {
  325. test_findFirstTokens() {
  326. const tokens = [
  327. new MDToken('Lorem', MDTokenType.Text),
  328. new MDToken(' ', MDTokenType.Whitespace),
  329. new MDToken('_', MDTokenType.Underscore),
  330. new MDToken('ipsum', MDTokenType.Text),
  331. new MDToken('_', MDTokenType.Underscore),
  332. new MDToken(' ', MDTokenType.Whitespace),
  333. new MDToken('dolor', MDTokenType.Text),
  334. new MDToken('_', MDTokenType.Underscore),
  335. new MDToken('sit', MDTokenType.Text),
  336. new MDToken('_', MDTokenType.Underscore),
  337. ];
  338. const pattern = [
  339. MDTokenType.Underscore,
  340. ];
  341. const result = MDToken.findFirstTokens(tokens, pattern);
  342. const expected = {
  343. tokens: [ tokens[2] ],
  344. index: 2,
  345. };
  346. this.assertEqual(JSON.stringify(result), JSON.stringify(expected));
  347. }
  348. test_findPairedTokens() {
  349. const tokens = [
  350. new MDToken('Lorem', MDTokenType.Text),
  351. new MDToken(' ', MDTokenType.Whitespace),
  352. new MDToken('_', MDTokenType.Underscore),
  353. new MDToken('ipsum', MDTokenType.Text),
  354. new MDToken('_', MDTokenType.Underscore),
  355. new MDToken(' ', MDTokenType.Whitespace),
  356. new MDToken('dolor', MDTokenType.Text),
  357. new MDToken('_', MDTokenType.Underscore),
  358. new MDToken('sit', MDTokenType.Text),
  359. new MDToken('_', MDTokenType.Underscore),
  360. ];
  361. const pattern = [
  362. MDTokenType.Underscore,
  363. ];
  364. const result = MDToken.findPairedTokens(tokens, pattern, pattern);
  365. const expected = {
  366. startTokens: [ tokens[2] ],
  367. contentTokens: [ tokens[3] ],
  368. endTokens: [ tokens[4] ],
  369. startIndex: 2,
  370. contentIndex: 3,
  371. endIndex: 4,
  372. totalLength: 3,
  373. }
  374. this.assertEqual(JSON.stringify(result), JSON.stringify(expected));
  375. }
  376. }
  377. class InlineTests extends BaseTest {
  378. /** @type {Markdown} */
  379. parser;
  380. md(markdown) {
  381. return normalizeWhitespace(this.parser.toHTML(markdown));
  382. }
  383. setUp() {
  384. this.parser = Markdown.completeParser;
  385. }
  386. test_simpleSingleParagraph() {
  387. let markdown = 'Lorem ipsum';
  388. let expected = '<p>Lorem ipsum</p>';
  389. let actual = this.md(markdown);
  390. this.assertEqual(actual, expected);
  391. }
  392. test_strong() {
  393. let markdown = 'Lorem **ipsum** dolor **sit**';
  394. let expected = '<p>Lorem <strong>ipsum</strong> dolor <strong>sit</strong></p>';
  395. let actual = this.md(markdown);
  396. this.assertEqual(actual, expected);
  397. }
  398. test_emphasis() {
  399. let markdown = 'Lorem _ipsum_ dolor _sit_';
  400. let expected = '<p>Lorem <em>ipsum</em> dolor <em>sit</em></p>';
  401. let actual = this.md(markdown);
  402. this.assertEqual(actual, expected);
  403. }
  404. test_strongEmphasis_cleanNesting1() {
  405. let markdown = 'Lorem **ipsum *dolor* sit** amet';
  406. let expected = '<p>Lorem <strong>ipsum <em>dolor</em> sit</strong> amet</p>';
  407. let actual = this.md(markdown);
  408. this.assertEqual(actual, expected);
  409. }
  410. test_strongEmphasis_cleanNesting2() {
  411. let markdown = 'Lorem *ipsum **dolor** sit* amet';
  412. let expected = '<p>Lorem <em>ipsum <strong>dolor</strong> sit</em> amet</p>';
  413. let actual = this.md(markdown);
  414. this.assertEqual(actual, expected);
  415. }
  416. test_strongEmphasis_tightNesting() {
  417. let markdown = 'Lorem ***ipsum*** dolor';
  418. let expected1 = '<p>Lorem <strong><em>ipsum</em></strong> dolor</p>';
  419. let expected2 = '<p>Lorem <em><strong>ipsum</strong></em> dolor</p>';
  420. let actual = this.md(markdown);
  421. this.assertTrue(actual == expected1 || actual == expected2);
  422. }
  423. test_strongEmphasis_lopsidedNesting1() {
  424. let markdown = 'Lorem ***ipsum* dolor** sit';
  425. let expected = '<p>Lorem <strong><em>ipsum</em> dolor</strong> sit</p>';
  426. let actual = this.md(markdown);
  427. this.assertEqual(actual, expected);
  428. }
  429. test_strongEmphasis_lopsidedNesting2() {
  430. let markdown = 'Lorem ***ipsum** dolor* sit';
  431. let expected = '<p>Lorem <em><strong>ipsum</strong> dolor</em> sit</p>';
  432. let actual = this.md(markdown);
  433. this.assertEqual(actual, expected);
  434. }
  435. test_strongEmphasis_lopsidedNesting3() {
  436. let markdown = 'Lorem **ipsum *dolor*** sit';
  437. let expected = '<p>Lorem <strong>ipsum <em>dolor</em></strong> sit</p>';
  438. let actual = this.md(markdown);
  439. this.assertEqual(actual, expected);
  440. }
  441. test_strongEmphasis_lopsidedNesting4() {
  442. let markdown = 'Lorem *ipsum **dolor*** sit';
  443. let expected = '<p>Lorem <em>ipsum <strong>dolor</strong></em> sit</p>';
  444. let actual = this.md(markdown);
  445. this.assertEqual(actual, expected);
  446. }
  447. test_inlineCode() {
  448. let markdown = 'Lorem `ipsum` dolor';
  449. let expected = '<p>Lorem <code>ipsum</code> dolor</p>';
  450. let actual = this.md(markdown);
  451. this.assertEqual(actual, expected);
  452. }
  453. test_inlineCode_withInnerBacktick() {
  454. let markdown = 'Lorem ``ip`su`m`` dolor';
  455. let expected = '<p>Lorem <code>ip`su`m</code> dolor</p>';
  456. let actual = this.md(markdown);
  457. this.assertEqual(actual, expected);
  458. }
  459. test_strikethrough_single() {
  460. let markdown = 'Lorem ~ipsum~ dolor';
  461. let expected = '<p>Lorem <strike>ipsum</strike> dolor</p>';
  462. let actual = this.md(markdown);
  463. this.assertEqual(actual, expected);
  464. }
  465. test_strikethrough_double() {
  466. let markdown = 'Lorem ~~ipsum~~ dolor';
  467. let expected = '<p>Lorem <strike>ipsum</strike> dolor</p>';
  468. let actual = this.md(markdown);
  469. this.assertEqual(actual, expected);
  470. }
  471. test_link_fullyQualified() {
  472. let markdown = 'Lorem [ipsum](https://example.com/path/page.html) dolor';
  473. let expected = '<p>Lorem <a href="https://example.com/path/page.html">ipsum</a> dolor</p>';
  474. let actual = this.md(markdown);
  475. this.assertEqual(actual, expected);
  476. }
  477. test_link_relative() {
  478. let markdown = 'Lorem [ipsum](page.html) dolor';
  479. let expected = '<p>Lorem <a href="page.html">ipsum</a> dolor</p>';
  480. let actual = this.md(markdown);
  481. this.assertEqual(actual, expected);
  482. }
  483. test_link_title() {
  484. let markdown = 'Lorem [ipsum](page.html "link title") dolor';
  485. let expected = '<p>Lorem <a href="page.html" title="link title">ipsum</a> dolor</p>';
  486. let actual = this.md(markdown);
  487. this.assertEqual(actual, expected);
  488. }
  489. test_link_literal() {
  490. let markdown = 'Lorem <https://example.com> dolor';
  491. let expected = '<p>Lorem <a href="https://example.com">https://example.com</a> dolor</p>';
  492. let actual = this.md(markdown);
  493. this.assertEqual(actual, expected);
  494. }
  495. test_link_ref() {
  496. let markdown = "Lorem [ipsum][ref] dolor\n\n[ref]: https://example.com";
  497. let expected = '<p>Lorem <a href="https://example.com">ipsum</a> dolor</p>';
  498. let actual = this.md(markdown);
  499. this.assertEqual(actual, expected);
  500. }
  501. test_link_email() {
  502. let markdown = 'Lorem [ipsum](user@example.com) dolor';
  503. let expected = '<p>Lorem <a href="mailto:&#117;&#115;&#101;&#114;&#64;&#101;&#120;&#97;&#109;&#112;&#108;&#101;&#46;&#99;&#111;&#109;">ipsum</a> dolor</p>';
  504. let actual = this.md(markdown);
  505. this.assertEqual(actual, expected);
  506. }
  507. test_link_email_withTitle() {
  508. let markdown = 'Lorem [ipsum](user@example.com "title") dolor';
  509. let expected = '<p>Lorem <a href="mailto:&#117;&#115;&#101;&#114;&#64;&#101;&#120;&#97;&#109;&#112;&#108;&#101;&#46;&#99;&#111;&#109;" title="title">ipsum</a> dolor</p>';
  510. let actual = this.md(markdown);
  511. this.assertEqual(actual, expected);
  512. }
  513. test_link_literalEmail() {
  514. let markdown = 'Lorem <user@example.com> dolor';
  515. let expected = '<p>Lorem <a href="mailto:&#117;&#115;&#101;&#114;&#64;&#101;&#120;&#97;&#109;&#112;&#108;&#101;&#46;&#99;&#111;&#109;">&#117;&#115;&#101;&#114;&#64;&#101;&#120;&#97;&#109;&#112;&#108;&#101;&#46;&#99;&#111;&#109;</a> dolor</p>';
  516. let actual = this.md(markdown);
  517. this.assertEqual(actual, expected);
  518. }
  519. test_image() {
  520. let markdown = 'Lorem ![alt text](image.jpg) dolor';
  521. let expected = '<p>Lorem <img src="image.jpg" alt="alt text"> dolor</p>';
  522. let actual = this.md(markdown);
  523. this.assertEqual(actual, expected);
  524. }
  525. test_image_noAlt() {
  526. let markdown = 'Lorem ![](image.jpg) dolor';
  527. let expected = '<p>Lorem <img src="image.jpg"> dolor</p>';
  528. let actual = this.md(markdown);
  529. this.assertEqual(actual, expected);
  530. }
  531. test_image_withTitle() {
  532. let markdown = 'Lorem ![alt text](image.jpg "image title") dolor';
  533. let expected = '<p>Lorem <img src="image.jpg" alt="alt text" title="image title"> dolor</p>';
  534. let actual = this.md(markdown);
  535. this.assertEqual(actual, expected);
  536. }
  537. }
  538. class BlockTests extends BaseTest {
  539. /** @type {Markdown} */
  540. parser;
  541. md(markdown) {
  542. return normalizeWhitespace(this.parser.toHTML(markdown));
  543. }
  544. setUp() {
  545. this.parser = Markdown.completeParser;
  546. }
  547. test_paragraphs() {
  548. let markdown = "Lorem ipsum\n\nDolor sit amet";
  549. let expected = "<p>Lorem ipsum</p> <p>Dolor sit amet</p>";
  550. let actual = this.md(markdown);
  551. this.assertEqual(actual, expected);
  552. }
  553. test_paragraph_lineGrouping() {
  554. let markdown = "Lorem ipsum\ndolor sit amet";
  555. let expected = "<p>Lorem ipsum dolor sit amet</p>";
  556. let actual = this.md(markdown);
  557. this.assertEqual(actual, expected);
  558. }
  559. test_unorderedList() {
  560. let markdown = "* Lorem\n* Ipsum\n* Dolor";
  561. let expected = '<ul> <li>Lorem</li> <li>Ipsum</li> <li>Dolor</li> </ul>';
  562. let actual = this.md(markdown);
  563. this.assertEqual(actual, expected);
  564. }
  565. test_orderedList() {
  566. let markdown = "1. Lorem\n1. Ipsum\n5. Dolor";
  567. let expected = '<ol start="1"> <li>Lorem</li> <li>Ipsum</li> <li>Dolor</li> </ol>';
  568. let actual = this.md(markdown);
  569. this.assertEqual(actual, expected);
  570. }
  571. test_orderedList_numbering() {
  572. let markdown = "4. Lorem\n1. Ipsum\n9. Dolor";
  573. let expected = '<ol start="4"> <li>Lorem</li> <li>Ipsum</li> <li>Dolor</li> </ol>';
  574. let actual = this.md(markdown);
  575. this.assertEqual(actual, expected);
  576. }
  577. test_blockquote() {
  578. let markdown = '> Lorem ipsum dolor';
  579. let expected = '<blockquote> <p>Lorem ipsum dolor</p> </blockquote>';
  580. let actual = this.md(markdown);
  581. this.assertEqual(actual, expected);
  582. }
  583. }
  584. </script>
  585. </head>
  586. <body>
  587. <div id="results"></div>
  588. </body>
  589. </html>