Merge commit '1ca76ee10d' into HEAD
This commit is contained in:
commit
10a2e6a01d
|
@ -73,7 +73,7 @@ function ___to_hex_for_printf(str) {
|
||||||
return '<span style="display: block;">' + hex + '</span>';
|
return '<span style="display: block;">' + hex + '</span>';
|
||||||
}
|
}
|
||||||
function ___specialchars_and_colour(s) {
|
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) {
|
switch (c) {
|
||||||
case " ": return '<span class="space"> </span>';
|
case " ": return '<span class="space"> </span>';
|
||||||
case "\\": return '<span class="specialchar">\\\\</span>';
|
case "\\": return '<span class="specialchar">\\\\</span>';
|
||||||
|
@ -733,6 +733,23 @@ var ___script_log_header = '' +
|
||||||
/*+*/ '})(window.console);\n' +
|
/*+*/ '})(window.console);\n' +
|
||||||
/*+*/ '\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) {
|
function ___file_contents_to_graphview(filesystem, path_of_this_file, s) {
|
||||||
var gv = '';
|
var gv = '';
|
||||||
var s2 = null;
|
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 ___previous_directory_node_style = 'color = "#80c5c5", fontcolor = "#80c5c5", class = dimmed_previous_directory';
|
||||||
var ___directory_node_style = 'color = "#008b8b", fontcolor = "#008b8b"'; // darkcyan = #008b8b
|
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) {
|
function ___entry_to_graphview(previous_filesystem, filesystem, x) {
|
||||||
var gv = '';
|
var gv = '';
|
||||||
gv += ___quote_gv(x[0]) + '\n';
|
gv += ___quote_gv(x[0]) + '\n';
|
||||||
|
@ -783,6 +795,7 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) {
|
||||||
var components = x[0].split('/');
|
var components = x[0].split('/');
|
||||||
|
|
||||||
var shortname = components[components.length - 1];
|
var shortname = components[components.length - 1];
|
||||||
|
var type = null;
|
||||||
if (___is_hashed_object_path(x[0])) {
|
if (___is_hashed_object_path(x[0])) {
|
||||||
// var hash = components.slice(components.length-2).join('');
|
// var hash = components.slice(components.length-2).join('');
|
||||||
shortname = shortname.substr(0, 3) + '…';
|
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('/');
|
var parent = components.slice(0, components.length - 1).join('/');
|
||||||
if (parent != '') {
|
if (parent != '') {
|
||||||
if (filesystem.hasOwnProperty(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';
|
gv += ___quote_gv(parent) + ' -> ' + ___quote_gv(x[0]) + ' ['+___directory_edge_style+'];\n';
|
||||||
} else {
|
} else {
|
||||||
|
// if the parent directory was not created in the filesystem, show the full path
|
||||||
shortname = parent + '/' + shortname;
|
shortname = parent + '/' + shortname;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -800,7 +815,7 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) {
|
||||||
// Put a transparent background to make the nodes clickable.
|
// Put a transparent background to make the nodes clickable.
|
||||||
gv += ___quote_gv(x[0]) + ' [ style="filled", fillcolor="transparent" ]';
|
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]) + ' ]';
|
gv += ___quote_gv(x[0]) + ' [ tooltip = ' + ___quote_gv(x[0]) + ' ]';
|
||||||
|
|
||||||
var id = 'gv-' + (___global_unique_id++);
|
var id = 'gv-' + (___global_unique_id++);
|
||||||
|
@ -808,9 +823,9 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) {
|
||||||
|
|
||||||
if (x[1] === null) {
|
if (x[1] === null) {
|
||||||
if (shortname.length <= 2) {
|
if (shortname.length <= 2) {
|
||||||
shortname = shortname + '\ndir';
|
type = '(dir)';
|
||||||
} else {
|
} else {
|
||||||
shortname = shortname + '\ndirectory';
|
type = '(directory)';
|
||||||
}
|
}
|
||||||
if (previous_filesystem.hasOwnProperty(x[0])) {
|
if (previous_filesystem.hasOwnProperty(x[0])) {
|
||||||
// dim nodes that existed in the previous_filesystem
|
// dim nodes that existed in the previous_filesystem
|
||||||
|
@ -820,7 +835,7 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) {
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
var contents = ___file_contents_to_graphview(filesystem, x[0], x[1]);
|
var contents = ___file_contents_to_graphview(filesystem, x[0], x[1]);
|
||||||
shortname = shortname + '\n(' + contents.type + ')';
|
type = '(' + contents.type + ')';
|
||||||
gv += contents.gv;
|
gv += contents.gv;
|
||||||
if (previous_filesystem.hasOwnProperty(x[0])) {
|
if (previous_filesystem.hasOwnProperty(x[0])) {
|
||||||
// dim nodes that existed in the previous_filesystem
|
// dim nodes that existed in the previous_filesystem
|
||||||
|
@ -829,7 +844,7 @@ function ___entry_to_graphview(previous_filesystem, filesystem, x) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// shortname as a label
|
// 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 };
|
return { id:id, gv:gv };
|
||||||
}
|
}
|
||||||
|
|
39
index.html
39
index.html
|
@ -1191,30 +1191,36 @@ This is done by creating a <em>tree</em> object</p>
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<p>In the contents of a tree, subdirectories (trees) are listed before files (blobs);
|
<p>In the contents of a tree, the entries are ordered alphabetically.</p>
|
||||||
within each group the entries are ordered alphabetically.</p>
|
|
||||||
|
|
||||||
<textarea id="in8">
|
<textarea id="in8">
|
||||||
// base_directory is a string
|
// base_directory is a string
|
||||||
// filenames is a list of strings
|
// filenames is a list of strings
|
||||||
// subtrees is a list of {name, hash} objects.
|
// subtrees is a list of {name, hash} objects.
|
||||||
function store_tree(base_directory, filenames, subtrees) {
|
function store_tree(base_directory, filenames, subtrees) {
|
||||||
function get_file_hash(filename) {
|
// entries will contain {name:'…', entry:'…'} objects
|
||||||
var path = join_paths(base_directory, filename);
|
var entries = [];
|
||||||
var hash = hash_object(true, 'blob', false, path)
|
|
||||||
return hex_to_raw_bytes(hash);
|
for (var i = 0; i < filenames.length; i++) {
|
||||||
|
var path = join_paths(base_directory, filenames[i]);
|
||||||
|
var hash = hash_object(true, 'blob', false, path);
|
||||||
|
var entry = "100644 " + filenames[i] + "\0" + hex_to_raw_bytes(hash);
|
||||||
|
entries.push({ name: filenames[i], entry: entry });
|
||||||
}
|
}
|
||||||
|
|
||||||
var blobs = filenames.map(function (filename) {
|
for (var j = 0; j < subtrees.length; j++) {
|
||||||
return "100644 " + filename + "\0" + get_file_hash(filename);
|
var entry = "40000 " + subtrees[j].name + "\0" + hex_to_raw_bytes(subtrees[j].hash);
|
||||||
|
entries.push({ name: subtrees[j], entry: entry });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sort the entries by name, alphabetically.
|
||||||
|
// Note that this won't work with e.g. unicode names.
|
||||||
|
entries.sort(function (a,b) {
|
||||||
|
return (a.name < b.name ? -1 : (a.name > b.name ? 1 : 0));
|
||||||
});
|
});
|
||||||
|
|
||||||
var trees = subtrees.map(function (subtree) {
|
// concatenate the entries
|
||||||
return "40000 " + subtree.name + "\0" + hex_to_raw_bytes(subtree.hash);
|
var tree_content = entries.map(function (entry) { return entry.entry; }).join('');
|
||||||
});
|
|
||||||
|
|
||||||
// blobs are listed before subtrees
|
|
||||||
var tree_content = blobs.join('') + trees.join('');
|
|
||||||
|
|
||||||
// cat tree_content | git hash-object -w -t tree --stdin
|
// cat tree_content | git hash-object -w -t tree --stdin
|
||||||
return hash_object(true, 'tree', true, tree_content);
|
return hash_object(true, 'tree', true, tree_content);
|
||||||
|
@ -2428,12 +2434,15 @@ commands.</p>
|
||||||
</section>
|
</section>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section id="changelog-errata">
|
||||||
<h2>Changelog and errata</h2>
|
<h2>Changelog and errata</h2>
|
||||||
<dl>
|
<dl>
|
||||||
<dt>v1</dt><dd>Initial version.</dd>
|
<dt>v1</dt><dd>Initial version.</dd>
|
||||||
<dt>v1.0.1</dt><dd>Internal changes to provide IPFS links.</dd>
|
<dt>v1.0.1</dt><dd>Internal changes to provide IPFS links.</dd>
|
||||||
<dt>v1.0.2</dt><dd>Added a sitemap for download tools.</dd>
|
<dt>v1.0.2</dt><dd>Added a sitemap for download tools.</dd>
|
||||||
|
<dt>v1.1.0</dt><dd>The <a href="storing-trees">section on storing tres</a> used to indicate that subtrees appear before blobs in the
|
||||||
|
binary representation of a tree. This was incorrect, the entries are simply sorted alphabetically without any
|
||||||
|
consideration of their type. Thanks to Exe for spotting this. Internal change: escaped some strings.</dd>
|
||||||
</dl>
|
</dl>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user