';
if (!omit_graph) { gv.js(); }
}
@@ -907,59 +970,60 @@ function ___git_eval(current) {
document.getElementById('hide-eval-' + current).style.display = '';
var script = ___script_log_header;
script += 'try {';
- for (i = 0; i <= current - 1; i++) {
+ for (var i = 0; i <= current - 1; i++) {
script += ___textarea_value(___global_editors[i]);
}
- script += '\n'
- + 'var ___previous_filesystem = {};\n'
- + 'for (k in filesystem) { ___previous_filesystem[k] = filesystem[k]; }\n'
- + '___log = [];\n';
+ script += '\n' +
+ /*+*/ 'var ___previous_filesystem = {};\n' +
+ /*+*/ 'for (k in filesystem) { ___previous_filesystem[k] = filesystem[k]; }\n' +
+ /*+*/ '___log = [];\n';
script += ___textarea_value(___global_editors[current]);
- script += '\n'
- + '} catch (e) {'
- + ' if (("" + e.message).indexOf("GIT: assertion failed: ") != 0) {'
- + ' throw e;'
- + ' } else {'
- + ' ___log.push({ alert: true, txt: "command failed" });'
- + ' }'
- + '}'
- + '"End of the script";\n'
- + '\n'
- + '\n'
- + '___eval_result_to_html("out" + current, filesystem, ___previous_filesystem, ___log, false);\n'
- + 'filesystem;\n';
+ script += '\n' +
+ /*+*/ '} catch (e) {' +
+ /*+*/ ' if (("" + e.message).indexOf("GIT: assertion failed: ") != 0) {' +
+ /*+*/ ' throw e;' +
+ /*+*/ ' } else {' +
+ /*+*/ ' ___log.push({ alert: true, txt: "command failed" });' +
+ /*+*/ ' }' +
+ /*+*/ '}' +
+ /*+*/ '"End of the script";\n' +
+ /*+*/ '\n' +
+ /*+*/ '\n' +
+ /*+*/ '___eval_result_to_html("out" + current, filesystem, ___previous_filesystem, ___log, false);\n' +
+ /*+*/ 'filesystem;\n';
try {
- eval(script);
+ /* jslint evil: true */ eval(script); /* jslint evil: false */
} catch (e) {
// Stack traces usually include :line:column
var rx = /:([0-9][0-9]*):[0-9][0-9]*/g;
var linecol = rx.exec(''+e.stack);
var line = null;
if (linecol && linecol.length > 0) {
- line=parseInt(linecol[1]);
+ line=parseInt(linecol[1], 10);
} else {
// Some older versions of Firefox and probably some other browsers use just :line
- var rx = /:([0-9][0-9]*)*/g;
- var justline = rx.exec(''+e.stack);
+ var rx2 = /:([0-9][0-9]*)*/g;
+ var justline = rx2.exec(''+e.stack);
if (justline && justline.length > 0) {
line=parseInt(justline[1], 10);
}
}
+ var showline = null;
if (typeof(line) == 'number') {
var lines = script.split('\n');
if (line < lines.length) {
var from = Math.max(0, line-2);
var to = Math.min(lines.length - 1, line+2+1);
- var showline = ''
- + 'Possible location of the error: near line ' + line + '\n'
- + '\n'
- + lines.slice(from, to).map(function(l, i) { return '' + (from + i) + ': ' + l; }).join('\n')
- + '\n'
- + '\n';
+ showline = '' +
+ /*+*/ 'Possible location of the error: near line ' + line + '\n' +
+ /*+*/ '\n' +
+ /*+*/ lines.slice(from, to).map(function(l, i) { return '' + (from + i) + ': ' + l; }).join('\n') +
+ /*+*/ '\n' +
+ /*+*/ '\n';
}
} else {
- var showline = 'Sorry, this tutorial could not pinpoint precisely\nthe location of the error.\n'
- + 'The stacktrace below may contain more information.\n'
+ showline = 'Sorry, this tutorial could not pinpoint precisely\nthe location of the error.\n' +
+ /*+*/ 'The stacktrace below may contain more information.\n';
}
var error = ___specialchars("" + e + "\n\n" + e.stack);
document.getElementById('out' + current).innerHTML = '
' + showline + error + '
';
@@ -980,7 +1044,7 @@ function ___process_elements() {
for (var i = 0; i < sections.length; i++) {
var level = ___level(sections[i]);
while (level < previousLevel) {
- var p = stack.pop();
+ stack.pop();
previousLevel--;
}
while (level > previousLevel) {
@@ -998,8 +1062,11 @@ function ___sections_to_html(sections) {
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 headers = sections[i].s.querySelectorAll('h2,h3');
+ if (!headers || headers.length < 1) {
+ if (window.console && window.console.log) { window.console.log("internal error: found no headers in section"); }
+ continue;
+ }
var target = sections[i].s.getAttribute('id');
var a = document.createElement('a');
li.appendChild(a);
@@ -1007,10 +1074,12 @@ function ___sections_to_html(sections) {
if (target) { a.setAttribute('href', '#' + target); }
if (target) {
var a2 = document.createElement('a');
- ___insertAfter(a2, headers[0]);
- a2.className = "permalink"
+ var hd = headers[0];
+ hd.parentElement.replaceChild(a2, hd);
+ a2.appendChild(hd);
+ a2.className = "permalink";
a2.setAttribute('href', '#' + target);
- a2.innerText = "🔗"
+ //a2.innerHTML += "🔗"
}
li.appendChild(___functions_to_html(sections[i].s));
li.appendChild(___sections_to_html(sections[i].subsections));
@@ -1037,7 +1106,7 @@ function ___functions_to_html(section) {
// Since CodeMirror replaces the textareas, the collection of HTML nodes
// is automatically updated in some browsers, and the indices become wrong
// after a replacement, so we copy the HTML element collection to a proper array.
- for (var j = 0; j < tas.length; j++) { ta.push(tas[j]); }
+ for (var i = 0; i < tas.length; i++) { ta[i] = tas[i]; }
for (var j = 0; j < ta.length; j++) {
if (___ancestor(ta[j], 'section') == section) {
var lines = ta[j].value.split('\n');
@@ -1045,17 +1114,19 @@ function ___functions_to_html(section) {
var editor = ret.editor;
var editor_id = ret.editor_id;
editor.on('keydown', ___clearScrolledToLine);
- for (var i = 0; i < lines.length; i++) {
+ for (var k = 0; k < lines.length; k++) {
var text = false;
- var fun = lines[i].match(/^function\s+([a-zA-Z_][a-zA-Z0-9_]*)/);
+ var fun = lines[k].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*=/);
+ var v = lines[k].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);');
+ /* split the javascript: … string to prevent jslint from complaining. */
+ var js = 'java'+'script';
+ a.setAttribute('href', js + ': ___scrollToLine(___global_editors['+(editor_id)+'], '+k+'); void(0);');
var code = document.createElement('code');
if (fun) {
var spanFunction = document.createElement('span');
@@ -1130,7 +1201,7 @@ function ___hide_eval(editor_id) {
function ___get_all_code() {
var all = '';
for (var i = 0; i < ___global_editors.length; i++) {
- var val = ___global_editors[i].getValue()
+ 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*/);
@@ -1145,7 +1216,7 @@ function ___copy_all_code() {
elem.innerHTML = '';
elem.appendChild(elem2);
var all_code = ___get_all_code();
- elem2.value = all_code
+ elem2.value = all_code;
elem2.focus();
elem2.disabled = false;
elem2.select();
@@ -1157,19 +1228,42 @@ function ___copy_all_code() {
function ___loc_count() {
var srclines = ___get_all_code().split('\n');
- var lcv = srclines.filter(function (l) { return ! (/^(\s*}?)?$/.test(l)); }).length
+ var lcv = srclines.filter(function (l) { return ! (/^(\s*}?)?$/.test(l)); }).length;
var lc = document.getElementsByClassName('loc-count');
for (var i = 0; i < lc.length; i++) {
lc[i].innerText = lcv;
}
var lctv = srclines.length;
var lct = document.getElementsByClassName('loc-count-total');
- for (var i = 0; i < lct.length; i++) {
- lct[i].innerText = lctv;
+ for (var j = 0; j < lct.length; j++) {
+ lct[j].innerText = lctv;
}
}
function ___git_tutorial_onload() {
___process_elements();
___loc_count();
-}
\ No newline at end of file
+}
+
+/* remove jslint "unused variable" warnings for these */
+var ___jslint_variables_called_from_event_handlers_and_from_tutorial = [
+ ___scroll_to_dest,
+ ___lolite,
+ ___deflated_click,
+ ___copyzip_click,
+ ___copyprintf_click,
+ ___click_graphview_hover,
+ ___mouseout_graphview_hover,
+ ___mouseover_graphview_hover,
+ ___eval_result_to_html,
+ ___git_eval,
+ ___scrollToLine,
+ ___hide_eval,
+ ___copy_all_code,
+ ___git_tutorial_onload,
+ sha1_from_bytes_returns_hex,
+ inflate,
+ deflate
+];
+// also ignore this dummy variable.
+___jslint_variables_called_from_event_handlers_and_from_tutorial = ___jslint_variables_called_from_event_handlers_and_from_tutorial;
\ No newline at end of file
diff --git a/index.html b/index.html
index d04b6c8..5e57151 100644
--- a/index.html
+++ b/index.html
@@ -1,3 +1,4 @@
+
@@ -41,12 +42,12 @@ function ___example(id, f) {
-
Git tutorial: reimplementing part of GIT in JavaScript
This article was written as part of my work for LIGO.
The main reference for this tutorial is the Pro Git book section on GIT internals.
@@ -148,7 +149,7 @@ Copyright notice:
(C) 1995-2013 Jean-loup Gailly and Mark Adler
-Copyright (c) <''year''> <''copyright holders''>
+Copyright (c) <''year''> <''copyright holders''>
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
@@ -237,7 +238,7 @@ GPL version 3
GNU GENERAL PUBLIC LICENSE
Version 3, 29 June 2007
- Copyright (C) 2007 Free Software Foundation, Inc.
+ Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
Everyone is permitted to copy and distribute verbatim copies
of this license document, but changing it is not allowed.
@@ -855,6 +856,7 @@ Program, unless a warranty or assumption of liability accompanies a
copy of the Program in return for a fee.
END OF TERMS AND CONDITIONS
+
@@ -887,7 +889,7 @@ vulnerabilities (user input is not sanitized when displayed).
-
Introduction
+
Introduction
GIT is based on a simple model, with a lot of shorthands for common
use cases. This model is sometimes hard to guess just from the
@@ -900,10 +902,10 @@ excluded, a few more in total.
-
The Operating System's filesystem
+
The Operating System's filesystem
-
Model of the filesystem
+
Model of the filesystem
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
@@ -916,7 +918,7 @@ var current_directory = '';
The filesystem exposes functions to read an entire file, create or
replace an entire file, create a directory, test the existence of a filesystem entry, and change the current directory.
-
Filesystem access functions (listdir)
+
Filesystem access functions (listdir)
It will be handy for some operations to list the contents of a
directory.
-
Example working tree
+
Example working tree
Our imaginary user will create a proj directory,
and start filling in some files.
-
+
A working tree designates the directory (and the subdirectories and files within) in which
the user will normally view and edit the files. GIT has commands to save the state of the working tree
(git commit), in order to be able to go back in time later on, and view older versions of the files.
@@ -1007,7 +1009,7 @@ write('proj/src/main.scm', '(map (lambda (x) (+ x 1)) (list 1 2 3))\n');
-
git init (creating .git)
+
git init (creating .git)
The first thing to do is to initialize the GIT directory.
For now, only the .git folder is needed, The rest
of the function implementing git init will be
@@ -1029,7 +1031,7 @@ git_init_mkdir();
-
git hash-object (storing a copy of a file in .git)
+
git hash-object (storing a copy of a file in .git)
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,
@@ -1106,7 +1108,7 @@ function hash_object(must_write, type, is_data, path_or_data) {
-
Adding a file to the GIT database
+
Adding a file to the GIT database
So far, our GIT database does not know about any of the user's
files. In order to add the contents of the README file in
the database, we use git hash-object -w -t blob README,
@@ -1132,7 +1134,7 @@ hash_object(true, 'blob', false, 'src/main.scm');
-
zlib compression
+
zlib compression
GIT compresses objects with zlib. The deflate() function used in
the script above comes from the pako 2.0.3 library.
To view a zlib-compressed object in your *nix terminal, simply write this
@@ -1150,7 +1152,7 @@ unzlib() {
-
Storing trees (list of hashed files and subtrees)
+
Storing trees (list of hashed files and subtrees)
At this point GIT knows about the contents of both of the user's
files, but it would be nice to also store the filenames.
This is done by creating a tree object
@@ -1214,7 +1216,7 @@ function hex_to_raw_bytes(hex) {
-
Example use of store_tree()
+
Example use of store_tree()
The following code, once uncommented, stores into the GIT database the trees for src
and for the root directory of the GIT project.
@@ -1229,7 +1231,7 @@ the hierarchy, and stores the corresponding trees bottom-up.
-
Storing a tree from a list of paths
+
Storing a tree from a list of paths
Making trees out of the subfolders one by one is cumbersome.
The following utility function takes a list of paths, and builds
a tree from those.
Now that the GIT database contains the entire tree for the current version,
a commit can be created. A commit contains
@@ -1368,7 +1370,7 @@ function format_timezone(tm) {
-
Storing an example commit
+
Storing an example commit
It is now possible to store a commit in the database. This saves
a copy of the tree along with some metadata about this version.
The first commit has no parent, which is represented by passing
@@ -1392,7 +1394,7 @@ var initial_commit = store_commit(
-
resolving references
+
resolving references
The next few subsections will introduce symbolic references
and other references like branch names, the special name HEAD
@@ -1468,7 +1470,7 @@ function trim_newline(s) {
-
git symbolic-ref
+
git symbolic-ref
git symbolic-ref is a low-level command which reads
(and in the official GIT implementation also writes and updates)
symbolic references given a path relative to .git/.
@@ -1537,7 +1539,7 @@ function git_symbolic_ref(ref) {
-
git rev-parse
+
git rev-parse
git rev-parse is another low-level command. It takes a symbolic reference or other reference,
and returns the hash. The difference with git symbolic-ref is that symbolic-ref follows indirections
to other references, and returns the last named reference in the chain of indirections, whereas rev-parse
@@ -1577,7 +1579,7 @@ function git_rev_parse(ref) {
-
git branch
+
git branch
A branch is a pointer to a commit, stored in a file in .git/refs/heads/name_of_the_branch.
The branch can be overwritten with git branch -f. Also, as will be explained later,
@@ -1649,7 +1651,7 @@ git_branch('main', initial_commit, true);
-
HEAD
+
HEAD
The HEAD indicates the "current" commit. It is set at first as part of the git init routine.
@@ -1682,8 +1684,8 @@ git_init_head();
Since the HEAD is supposed to be a transient pointer, it is easy to lose track of the hash of
an important commit. For example, the following sequence of operations:
-
-
+
+
git checkout 0123456789abcdef0123456789abcdef01234567
touch new_file
@@ -1691,11 +1693,11 @@ git add new_file
git commit -m 'This is a commit adding a new file'
git checkout branch-of-feature-foobar
-
+
- roughly means:
+
roughly means:
-
+
HEAD = 0123456789abcdef0123456789abcdef01234567
// overwrite the contents of the working tree with
// the contents of commit 0123456789abcdef0123456789abcdef01234567
@@ -1706,8 +1708,7 @@ HEAD = commit(…)
// Checkout other branch
HEAD = git_rev_parse('branch-of-feature-foobar')
-
-
+
The hash of the new commit which is stored in HEAD on the second step is overwritten
@@ -1746,12 +1747,12 @@ HEAD = git_rev_parse('branch-of-feature-foobar')
-
git config
-
- The official implementation of GIT stores the settings in various files (.git/config within a repository,
- ~/.gitconfig in the user's home folder, and several other places).
-
-
-
git commit
+
git commit
The git commit command stores a commit (metadata and a pointer to a tree
@@ -1857,7 +1858,7 @@ var second_commit = git_commit(['README', 'src/main.scm'], 'Some updates');
-
git tag
+
git tag
Tags behave like branches, but are stored in .git/refs/tags/the_tag_name
and a tag is not normally modified. Once created, it's supposed to always point
to the same version.
The git checkout commit-hash-or-reference command modifies the HEAD to point to the given commit,
and modifies the working tree to match the contents of the tree object pointed to by that commit.
@@ -1920,7 +1921,7 @@ function git_checkout(tag_or_branch_or_hash) {
}
-
Checkout, branches and other references
+
Checkout, branches and other references
The HEAD does not normally point to a tag. Although nothing actually
prevents writing ref: refs/tags/v1.0 into .git/HEAD, the GIT
commands will not automatically do this. For example, git checkout tag-or-branch-or-hash
@@ -1928,7 +1929,7 @@ function git_checkout(tag_or_branch_or_hash) {
-
Checking out files
+
Checking out files
In order to replace the contents of the working tree with those of the given commit, we
recursively compare the subtrees, deleting from the working tree the files or directories
@@ -1978,7 +1979,7 @@ function checkout_tree(path_prefix, hash) {
-
Assert
+
Assert
The checkout_tree() function needs to read the commit, tree and blob objects from the
.git/ folder. The following sections will introduce some parsers for these objects.
@@ -1992,7 +1993,7 @@ function assert(boolean, text) {
-
Reading compressed objects
+
Reading compressed objects
The GIT objects which are stored in .git/objects are compressed with zlib, and need to be
uncompressed before they can be parsed. The actual implementation of GIT also stores some objects in packs. Packs
contain a large number of objects, and used a form of delta compression, which effectively stores objects as the diff with
@@ -2021,7 +2022,7 @@ function parse_object(hash) {
-
Parsing tree objects
+
Parsing tree objects
We will start by parsing tree objects. As a reminder, a tree object has the following form: