';
+ if (! just_table) {
+ html += 'Filesystem contents: ' + entries.length + " files and directories. "
+ + ''
+ + "Copy commands to recreate in *nix terminal"
+ + "."
+ + " "
+ + '';
+ }
+ html += ___filesystem_to_table(fs, previous_filesystem).outerHTML // TODO: use DOM primitives instead.
+ + '
';
+ return html;
}
function ___textarea_value(elem) {
if (elem.getValue) {
@@ -389,10 +518,12 @@ var global_filesystem=false;
function ___git_eval(current) {
document.getElementById('hide-eval-' + current).style.display = '';
var script = '';
- for (i = 0; i <= current; i++) {
+ for (i = 0; i <= current - 1; i++) {
script += ___textarea_value(___global_editors[i]);
}
- script += "\n document.getElementById('out' + current).innerHTML = ___filesystem_to_string(filesystem); filesystem;";
+ script += "\n var ___previous_filesystem = {}; for (k in filesystem) { ___previous_filesystem[k] = filesystem[k]; }\n";
+ script += ___textarea_value(___global_editors[current]);
+ script += "\n document.getElementById('out' + current).innerHTML = ___filesystem_to_string(filesystem, false, ___previous_filesystem); filesystem;";
try {
global_filesystem = eval(script);
} catch (e) {
@@ -402,190 +533,188 @@ function ___git_eval(current) {
}
function ___level(s) {
- if (s) {
- return (s.tagName == 'SECTION' ? 1 : 0) + ___level(s.parentElement);
- } else {
- return 0;
- }
+ if (s) {
+ return (s.tagName == 'SECTION' ? 1 : 0) + ___level(s.parentElement);
+ } else {
+ return 0;
}
- function ___process_elements() {
- var sections = document.getElementsByTagName('section');
- var stack = [[]];
- var previousLevel = 1;
- for (var i = 0; i < sections.length; i++) {
- var level = ___level(sections[i]);
- while (level < previousLevel) {
- var p = stack.pop();
- previousLevel--;
- }
- while (level > previousLevel) {
- var top_of_stack = stack[stack.length-1];
- stack.push(top_of_stack[top_of_stack.length-1].subsections);
- previousLevel++;
- }
- stack[stack.length-1].push({ s: sections[i], subsections: [] });
+}
+function ___process_elements() {
+ var sections = document.getElementsByTagName('section');
+ var stack = [[]];
+ var previousLevel = 1;
+ for (var i = 0; i < sections.length; i++) {
+ var level = ___level(sections[i]);
+ while (level < previousLevel) {
+ var p = stack.pop();
+ previousLevel--;
}
- var nested = stack[0];
- document.getElementById('toc').appendChild(___sections_to_html(nested));
- }
- function ___sections_to_html(sections) {
- var ol = document.createElement('ol');
- for (var i = 0; i < sections.length; i++) {
- var li = document.createElement('li');
- ol.appendChild(li);
- var headers = sections[i].s.getElementsByTagName('h1');
- console.assert(!headers || headers.length >= 1)
- var target = sections[i].s.getAttribute('id');
- var a = document.createElement('a');
- li.appendChild(a);
- a.innerHTML = headers[0].innerHTML;
- if (target) { a.setAttribute('href', '#' + target); }
- if (target) {
- var a2 = document.createElement('a');
- ___insertAfter(a2, headers[0]);
- a2.className = "permalink"
- a2.setAttribute('href', '#' + target);
- a2.innerText = "🔗"
- }
- li.appendChild(___functions_to_html(sections[i].s));
- li.appendChild(___sections_to_html(sections[i].subsections));
+ while (level > previousLevel) {
+ var top_of_stack = stack[stack.length-1];
+ stack.push(top_of_stack[top_of_stack.length-1].subsections);
+ previousLevel++;
}
- return ol;
+ stack[stack.length-1].push({ s: sections[i], subsections: [] });
}
- function ___insertAfter(elt, ref) {
- ref.parentElement.insertBefore(elt, ref.nextSibling);
- }
- function ___ancestor(elem, tag) {
- if (! elem) {
- return false;
+ var nested = stack[0];
+ document.getElementById('toc').appendChild(___sections_to_html(nested));
+}
+function ___sections_to_html(sections) {
+ var ol = document.createElement('ol');
+ for (var i = 0; i < sections.length; i++) {
+ var li = document.createElement('li');
+ ol.appendChild(li);
+ var headers = sections[i].s.getElementsByTagName('h1');
+ console.assert(!headers || headers.length >= 1)
+ var target = sections[i].s.getAttribute('id');
+ var a = document.createElement('a');
+ li.appendChild(a);
+ a.innerHTML = headers[0].innerHTML;
+ if (target) { a.setAttribute('href', '#' + target); }
+ if (target) {
+ var a2 = document.createElement('a');
+ ___insertAfter(a2, headers[0]);
+ a2.className = "permalink"
+ a2.setAttribute('href', '#' + target);
+ a2.innerText = "🔗"
}
- if (elem.tagName.toLowerCase() == tag) {
- return elem;
- }
- return ___ancestor(elem.parentElement, tag);
+ li.appendChild(___functions_to_html(sections[i].s));
+ li.appendChild(___sections_to_html(sections[i].subsections));
}
- var ___global_editors = [];
- function ___functions_to_html(section) {
- var ul = document.createElement('ul');
- var ta = section.getElementsByTagName('textarea');
- for (var j = 0; j < ta.length; j++) {
- if (___ancestor(ta[j], 'section') == section) {
- var lines = ta[j].value.split('\n');
-
- var ret = ___toCodeMirror(ta[j]);
- var editor = ret.editor;
- var editor_id = ret.editor_id;
-
- editor.on('keydown', ___clearScrolledToLine);
-
- for (var i = 0; i < lines.length; i++) {
- var text = false;
-
- var fun = lines[i].match(/^function\s+([a-zA-Z_][a-zA-Z0-9_]*)/);
- if (fun) { text = fun[1] + '()'; }
- var v = lines[i].match(/^var\s+([a-zA-Z_][a-zA-Z0-9_]*)\s*=/);
- if (v) { text = v[1]; }
-
- if (text) {
- var li = document.createElement('li');
- var a = document.createElement('a');
- a.setAttribute('href', 'javascript: ___scrollToLine(___global_editors['+(editor_id)+'], '+i+'); void(0);');
- var code = document.createElement('code');
+ return ol;
+}
+function ___insertAfter(elt, ref) {
+ ref.parentElement.insertBefore(elt, ref.nextSibling);
+}
+function ___ancestor(elem, tag) {
+ if (! elem) {
+ return false;
+ }
+ if (elem.tagName.toLowerCase() == tag) {
+ return elem;
+ }
+ return ___ancestor(elem.parentElement, tag);
+}
+var ___global_editors = [];
+function ___functions_to_html(section) {
+ var ul = document.createElement('ul');
+ var ta = section.getElementsByTagName('textarea');
+ for (var j = 0; j < ta.length; j++) {
+ if (___ancestor(ta[j], 'section') == section) {
+ var lines = ta[j].value.split('\n');
+ var ret = ___toCodeMirror(ta[j]);
+ var editor = ret.editor;
+ var editor_id = ret.editor_id;
+ editor.on('keydown', ___clearScrolledToLine);
+ for (var i = 0; i < lines.length; i++) {
+ var text = false;
+
+ var fun = lines[i].match(/^function\s+([a-zA-Z_][a-zA-Z0-9_]*)/);
+ if (fun) { text = fun[1]; }
+ var v = lines[i].match(/^var\s+([a-zA-Z_][a-zA-Z0-9_]*)\s*=/);
+ if (v) { text = v[1]; }
+ if (text) {
+ var li = document.createElement('li');
+ var a = document.createElement('a');
+ a.setAttribute('href', 'javascript: ___scrollToLine(___global_editors['+(editor_id)+'], '+i+'); void(0);');
+ var code = document.createElement('code');
+ if (fun) {
+ var spanFunction = document.createElement('span');
+ spanFunction.className = 'function';
+ spanFunction.innerText = text;
+ var parens = document.createTextNode('()');
+ code.appendChild(spanFunction);
+ code.appendChild(parens);
+ } else {
+ code.className = 'assignment';
code.innerText = text;
- a.appendChild(code);
- li.appendChild(a);
- ul.appendChild(li);
}
+ a.appendChild(code);
+ li.appendChild(a);
+ ul.appendChild(li);
}
}
}
- return ul;
}
- var ___global_current_highlighted_editor_and_line = false;
- function ___clearScrolledToLine() {
- var current = ___global_current_highlighted_editor_and_line;
- if (current) {
- current.editor.removeLineClass(current.line, 'background', 'scrolled-to-line');
- }
- ___global_current_highlighted_editor_and_line = false;
+ return ul;
+}
+var ___global_current_highlighted_editor_and_line = false;
+function ___clearScrolledToLine() {
+ var current = ___global_current_highlighted_editor_and_line;
+ if (current) {
+ current.editor.removeLineClass(current.line, 'background', 'scrolled-to-line');
}
- function ___scrollToLine(editor, line) {
- ___clearScrolledToLine();
- ___global_current_highlighted_editor_and_line = { editor: editor, line: line };
-
- editor.addLineClass(line, 'background', 'scrolled-to-line');
-
- var editorOffset = ___getOffset(editor.getScrollerElement()).top;
- var lineOffset = editor.charCoords({line: line, ch: 0}, "local").top;
- document.body.scrollTo(0, editorOffset + lineOffset - window.innerHeight/2);
+ ___global_current_highlighted_editor_and_line = false;
+}
+function ___scrollToLine(editor, line) {
+ ___clearScrolledToLine();
+ ___global_current_highlighted_editor_and_line = { editor: editor, line: line };
+ editor.addLineClass(line, 'background', 'scrolled-to-line');
+ var editorOffset = ___getOffset(editor.getScrollerElement()).top;
+ var lineOffset = editor.charCoords({line: line, ch: 0}, "local").top;
+ document.body.scrollTo(0, editorOffset + lineOffset - window.innerHeight/2);
+}
+function ___toCodeMirror(ta) {
+ var editor = CodeMirror.fromTextArea(ta, {
+ mode: 'javascript',
+ lineNumbers: true,
+ viewportMargin: Infinity
+ });
+ var id = ta.getAttribute('id');
+ ta.remove();
+ var wrapper = editor.getWrapperElement();
+ wrapper.setAttribute('id', id);
+ var editor_id = ___global_editors.length;
+ ___global_editors[editor_id] = editor;
+ var eval_button = document.createElement('input');
+ eval_button.setAttribute('type', 'button');
+ eval_button.setAttribute('value', 'eval');
+ eval_button.setAttribute('onclick', '___git_eval('+editor_id+')');
+ ___insertAfter(eval_button, wrapper);
+ var hide_eval_button = document.createElement('input');
+ hide_eval_button.setAttribute('id', 'hide-eval-' + editor_id);
+ hide_eval_button.setAttribute('type', 'button');
+ hide_eval_button.setAttribute('value', 'hide output');
+ hide_eval_button.setAttribute('onclick', '___hide_eval('+editor_id+')');
+ hide_eval_button.style.display = 'none';
+ ___insertAfter(hide_eval_button, eval_button);
+ var out_div = document.createElement('div');
+ out_div.setAttribute('id', 'out' + editor_id);
+ ___insertAfter(out_div, hide_eval_button);
+ return { editor: editor, editor_id: editor_id };
+}
+function ___hide_eval(editor_id) {
+ document.getElementById('out' + editor_id).innerHTML = '';
+ document.getElementById('hide-eval-' + editor_id).style.display = 'none';
+ ___hilite_off();
+}
+function ___get_all_code() {
+ var all = '';
+ for (var i = 0; i < ___global_editors.length; i++) {
+ var val = ___global_editors[i].getValue()
+ all += val + (val.endsWith('\n') ? '' : '\n') + (val.endsWith('\n\n') ? '' : '\n');
}
- function ___toCodeMirror(ta) {
- var editor = CodeMirror.fromTextArea(ta, {
- mode: 'javascript',
- lineNumbers: true,
- viewportMargin: Infinity
- });
- var id = ta.getAttribute('id');
- ta.remove();
- var wrapper = editor.getWrapperElement();
- wrapper.setAttribute('id', id);
-
- var editor_id = ___global_editors.length;
- ___global_editors[editor_id] = editor;
-
- var eval_button = document.createElement('input');
- eval_button.setAttribute('type', 'button');
- eval_button.setAttribute('value', 'eval');
- eval_button.setAttribute('onclick', '___git_eval('+editor_id+')');
- ___insertAfter(eval_button, wrapper);
-
- var hide_eval_button = document.createElement('input');
- hide_eval_button.setAttribute('id', 'hide-eval-' + editor_id);
- hide_eval_button.setAttribute('type', 'button');
- hide_eval_button.setAttribute('value', 'hide output');
- hide_eval_button.setAttribute('onclick', '___hide_eval('+editor_id+')');
- hide_eval_button.style.display = 'none';
- ___insertAfter(hide_eval_button, eval_button);
-
- var out_div = document.createElement('div');
- out_div.setAttribute('id', 'out' + editor_id);
- ___insertAfter(out_div, hide_eval_button);
-
- return { editor: editor, editor_id: editor_id };
+ return all.substr(0, all.length-1/*remove last newline in the last \n\n*/);
+}
+function ___copy_all_code() {
+ var elem = document.getElementById('copy-all-code');
+ if (elem.style.display != "none") {
+ elem.style.display = "none";
+ } else {
+ elem.style.display = '';
+ var elem2 = document.createElement('textarea');
+ elem.innerHTML = '';
+ elem.appendChild(elem2);
+ var all_code = ___get_all_code();
+ elem2.value = all_code
+ elem2.focus();
+ elem2.disabled = false;
+ elem2.select();
+ elem2.setSelectionRange(0, elem2.value.length * 10); // for mobile devices?
+ document.execCommand('copy');
+ elem2.disabled = true;
}
- function ___hide_eval(editor_id) {
- document.getElementById('out' + editor_id).innerHTML = '';
- document.getElementById('hide-eval-' + editor_id).style.display = 'none';
- ___hilite_off();
- }
- function ___get_all_code() {
- var all = '';
- for (var i = 0; i < ___global_editors.length; i++) {
- var val = ___global_editors[i].getValue()
- all += val + (val.endsWith('\n') ? '' : '\n') + (val.endsWith('\n\n') ? '' : '\n');
- }
- return all.substr(0, all.length-1/*remove last newline in the last \n\n*/);
- }
- function ___copy_all_code() {
- var elem = document.getElementById('copy-all-code');
- if (elem.style.display != "none") {
- elem.style.display = "none";
- } else {
- elem.style.display = '';
- var elem2 = document.createElement('textarea');
- elem.innerHTML = '';
- elem.appendChild(elem2);
- var all_code = ___get_all_code();
- elem2.value = all_code
- elem2.focus();
- elem2.disabled = false;
- elem2.select();
- elem2.setSelectionRange(0, elem2.value.length * 10); // for mobile devices?
- document.execCommand('copy');
- elem2.disabled = true;
- }
- }
- ___process_elements();
+}
function ___loc_count() {
var srclines = ___get_all_code().split('\n');
@@ -600,4 +729,8 @@ function ___loc_count() {
lct[i].innerText = lctv;
}
}
-___loc_count();
\ No newline at end of file
+
+function ___git_tutorial_onload() {
+ ___process_elements();
+ ___loc_count();
+}
\ No newline at end of file
diff --git a/index.html b/index.html
index e95ede9..d53b867 100644
--- a/index.html
+++ b/index.html
@@ -9,12 +9,31 @@
+
+
+
Under construction
-
The main reference for this tutorial is the Pro Git book section on GIT internals.
This tutorial uses three libraries:
@@ -24,8 +43,6 @@
pako 2.0.3, released under the MIT and Zlib licenses, see the project page for details.
-
-
Introduction
@@ -44,7 +61,7 @@ excluded, a few more in total.
Model of the filesystem
-
We will simulate the Operating System's filesystem with a very
+
The Operating System's filesystem will be simulated by a very
simple key-value store. In this very simple filesystem, directories
are entries mapped to null and files are entries mapped
to strings. The path to the current directory is stored in a separate
@@ -124,21 +141,70 @@ function join_paths(a, b) {
return (a == "") ? b : (a + "/" + b);
}
+// git init (partial implementation: create the .git directory)
function git_init_mkdir() {
mkdir(join_paths(current_directory, '.git'));
}
git_init_mkdir();
+
Click on the eval button to see the files and directories that were
+ created so far.
git hash-object (storing a copy of a file in .git)
-
The most basic element of a GIT repository is an object. It is a
-copy of a file that is stored in GIT's database. That copy is
-stored under a unique name. The unique name is obtained by hashing the
-contents of the file.
+
The most basic element of a GIT repository is an object. Objects have a type which can be
+blob (individual files), tree (directories),
+commit (pointers to a specific version of the root directory,
+with a description and some metadata) and tag (named pointers to a specific commit,
+with a description and some metadata).
+
+When a file is added to the git repostitory, a compressed copy is stored in GIT's database,
+in the .git/objects/ folder. This copy is a blob object.
+
The compressed copy is given a unique filename, which is obtained by hashing the contents of the original file.
+Some filesystems have poor performance when a single directory contains a large number of files, and some filesystems
+have a limit on the number of files that a directory may contain. To circumvent these issues, the first two characters
+of the hash are used as the name of an intermediate directory: if a file's hash is 0a1bd…, its compressed
+copy will be stored in .git/objects/0a/1bd…
+
+
This function creates a file that looks like this:
+
+
+
+
+
The objects stored in the GIT database are compressed with zlib
+(using the "deflate" compression method). The filesystem view shows
+the marker deflated: followed by the
+uncompressed data. Click on the (un)compressed data to toggle between
+this pretty-printed view and the raw compressed data.
+
+
When creating some blob objects, the result could be, for example:
+
+
+
+
+
This function reproduces faithfully the behaviour of (a subset of the options of)
+the git hash-object command which can be called on a real git command-line.
The objects stored in the GIT database are compressed with zlib
-(using the "deflate" compression method). The filesystem view shows
-the marker deflated: followed by the
-uncompressed data. Click on the file contents to toggle between this
-pretty-printed view and the raw compressed data.
-
+
Click on the eval button to see the file that was
+created by this call.
-
You will notice that the database does not contain the name of the
-file, only its contents, stored under a unique identifier which is
-derived by hashing its contents. Let's add the second user file
+
You can notice that the database does not contain the name of the
+original file, only its content, stored under a unique identifier which is
+derived by hashing that content. Let's add the second user file
to the database.