From c066e5015128c3cb134ac1245093df3a2e3aef17 Mon Sep 17 00:00:00 2001 From: Suzanne Soy Date: Thu, 8 Jul 2021 23:14:32 +0100 Subject: [PATCH] Escape strings in the graph description --- git-tutorial.js | 37 ++++++++++++++++++++++++++----------- 1 file changed, 26 insertions(+), 11 deletions(-) diff --git a/git-tutorial.js b/git-tutorial.js index fa9c959..2b86547 100644 --- a/git-tutorial.js +++ b/git-tutorial.js @@ -73,7 +73,7 @@ function ___to_hex_for_printf(str) { return '' + hex + ''; } function ___specialchars_and_colour(s) { - return s.replace(/[^-a-zA-Z0-9+_/!%$@.()':]/g, function (c) { + return s.replace(/[^-a-zA-Z0-9+_/!%$@.()':^]/g, function (c) { switch (c) { case " ": return ' '; case "\\": return '\\\\'; @@ -733,6 +733,23 @@ var ___script_log_header = '' + /*+*/ '})(window.console);\n' + /*+*/ '\n'; +function ___escape_gv(name) { + return name.replace(/[^- a-zA-Z0-9+_/!%$@.()':&<>'…^]/g, function (c) { + switch (c) { + case "\\": return '\\\\\\\\'; + case "\0": return '\\\\000'; + case "\r": return '\\\\r'; + case "\n": return '\\\\n'; + case "\t": return '\\\\t'; + case '"': return '\\"'; + default: return '\\\\x'+___left_pad(c.charCodeAt(0).toString(16), 0, 2)+''; + } + }); +} +function ___quote_gv(name) { + return '"' + ___escape_gv(name) + '"'; +} + function ___file_contents_to_graphview(filesystem, path_of_this_file, s) { var gv = ''; var s2 = null; @@ -771,11 +788,6 @@ var ___previous_file_node_style = 'color = "#808080", fontcolor = "#808080", cla var ___previous_directory_node_style = 'color = "#80c5c5", fontcolor = "#80c5c5", class = dimmed_previous_directory'; var ___directory_node_style = 'color = "#008b8b", fontcolor = "#008b8b"'; // darkcyan = #008b8b -function ___quote_gv(name) { - if (window.console && window.console.log) { window.console.log('TODO: escape GV'); } - return '"' + name.replace('\n', '\\n') + '"'; -} - function ___entry_to_graphview(previous_filesystem, filesystem, x) { var gv = ''; gv += ___quote_gv(x[0]) + '\n'; @@ -783,6 +795,7 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) { var components = x[0].split('/'); var shortname = components[components.length - 1]; + var type = null; if (___is_hashed_object_path(x[0])) { // var hash = components.slice(components.length-2).join(''); shortname = shortname.substr(0, 3) + '…'; @@ -791,8 +804,10 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) { var parent = components.slice(0, components.length - 1).join('/'); if (parent != '') { if (filesystem.hasOwnProperty(parent)) { + // show arrow from the parent to this element, if the parent directory exists in the filesystem. gv += ___quote_gv(parent) + ' -> ' + ___quote_gv(x[0]) + ' ['+___directory_edge_style+'];\n'; } else { + // if the parent directory was not created in the filesystem, show the full path shortname = parent + '/' + shortname; } } @@ -800,7 +815,7 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) { // Put a transparent background to make the nodes clickable. gv += ___quote_gv(x[0]) + ' [ style="filled", fillcolor="transparent" ]'; - // contents of the file as a tooltip: + // full name of the file as a tooltip: gv += ___quote_gv(x[0]) + ' [ tooltip = ' + ___quote_gv(x[0]) + ' ]'; var id = 'gv-' + (___global_unique_id++); @@ -808,9 +823,9 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) { if (x[1] === null) { if (shortname.length <= 2) { - shortname = shortname + '\ndir'; + type = '(dir)'; } else { - shortname = shortname + '\ndirectory'; + type = '(directory)'; } if (previous_filesystem.hasOwnProperty(x[0])) { // dim nodes that existed in the previous_filesystem @@ -820,7 +835,7 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) { } } else { var contents = ___file_contents_to_graphview(filesystem, x[0], x[1]); - shortname = shortname + '\n(' + contents.type + ')'; + type = '(' + contents.type + ')'; gv += contents.gv; if (previous_filesystem.hasOwnProperty(x[0])) { // dim nodes that existed in the previous_filesystem @@ -829,7 +844,7 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) { } // shortname as a label - gv += ___quote_gv(x[0]) + ' [ label = ' + ___quote_gv(shortname) + ' ]'; + gv += ___quote_gv(x[0]) + ' [ label = "' + ___escape_gv(shortname) + (type == null ? '' : '\\n' + ___escape_gv(type)) + '" ]'; return { id:id, gv:gv }; }