Commit 6cc22ee5 authored by Ivanq's avatar Ivanq

Merge branch 'gitwrite'

parents dfc752ac 31b22810
......@@ -37,6 +37,8 @@
<script type="text/javascript" src="../js/ZeroFS.js"></script>
<script type="text/javascript" src="../js/ZeroAuth.js"></script>
<script type="text/javascript" src="../js/ZeroDB.js"></script>
<script type="text/javascript" src="../js/pako.js"></script>
<script type="text/javascript" src="../js/sha.js"></script>
<script type="text/javascript" src="../js/git.js"></script>
<script type="text/javascript" src="../js/repo.js"></script>
<script type="text/javascript" src="../repo/js/user_common.js"></script>
......
......@@ -41,6 +41,20 @@ class Git {
return char;
}).join("");
}
packSha(str) {
let items = str.split("").map(char => {
if(char >= "0" && char <= "9") {
return char.charCodeAt(0) - "0".charCodeAt(0);
} else if(char >= "a" && char <= "z") {
return char.charCodeAt(0) - "a".charCodeAt(0) + 10;
}
});
let result = [];
for(let i = 0; i < items.length; i += 2) {
result.push(items[i] * 16 + items[i + 1]);
}
return result;
}
subArray(array, begin, length) {
if(length === undefined) {
return array.slice(begin);
......@@ -50,10 +64,19 @@ class Git {
}
appendArray(source, destination) {
source.forEach(item => destination.push(item));
return destination;
}
concat(...arrs) {
let destination = [];
arrs.forEach(arr => this.appendArray(arr, destination));
return destination;
}
arrayToString(array) {
return Array.from(array).map(char => String.fromCharCode(char)).join("");
}
stringToArray(string) {
return string.split("").map(char => char.charCodeAt(0));
}
isSha(str) {
return (
str.length == 40 &&
......@@ -78,9 +101,25 @@ class Git {
readDirectory(path, recursive) {
return this.zeroFS.readDirectory(this.root + "/" + path, recursive);
}
writeFile(path, content) {
return this.zeroFS.writeFile(this.root + "/" + path, Array.from(content).map(char => String.fromCharCode(char)).join(""), true);
}
inflate(string) {
return pako.inflate(string);
}
deflate(string) {
return pako.deflate(string);
}
sha(string) {
if(string instanceof Array) {
string = new Uint8Array(string);
}
let sha = new jsSHA("SHA-1", "ARRAYBUFFER");
sha.update(string);
return sha.getHash("HEX");
}
// Object commands
readObject(id) {
......@@ -100,6 +139,13 @@ class Git {
};
});
}
writeObject(type, content) {
let data = this.concat(this.stringToArray(type + " " + content.length), [0], content);
let id = this.sha(data);
return this.writeFile("objects/" + id.substr(0, 2) + "/" + id.substr(2), this.deflate(data))
.then(() => id);
}
// Packed objects
findPackedObjects() {
......@@ -353,7 +399,7 @@ class Git {
currentPos += 20;
items.push({
mode: mode,
type: mode.length == 6 && mode.indexOf("10") == 0 ? "blob" : "tree",
name: name,
id: objectId
});
......@@ -468,6 +514,81 @@ class Git {
});
}
// Object saving
writeBlob(content) {
return this.writeObject("blob", content);
}
writeTree(items) {
let content = [];
items.forEach(item => {
this.appendArray(this.concat(this.stringToArray((item.type == "tree" ? "040000" : "100644") + " " + item.name), [0], this.packSha(item.id)), content);
});
return this.writeObject("tree", content);
}
writeTreeRecursive(items) {
let content = items.map(item => {
if(item.type == "tree") {
if(item.id) {
// Use existing tree
return Promise.resolve({
type: "tree",
name: item.name,
id: item.id
});
}
return this.writeTreeRecursive(item.content)
.then(id => {
return {
type: "tree",
name: item.name,
id: id
};
});
} else if(item.type == "blob") {
if(item.id) {
// Use existing blob
return Promise.resolve({
type: "blob",
name: item.name,
id: item.id
});
}
return this.writeBlob(item.content)
.then(id => {
return {
type: "blob",
name: item.name,
id: id
};
});
}
});
return Promise.all(content)
.then(content => {
return this.writeTree(content);
});
}
writePlainCommit(commit) {
let content = "";
content += "tree " + commit.tree + "\n";
content += commit.parents.map(parent => "parent " + parent + "\n").join("");
content += "author " + commit.author + "\n";
content += "committer " + commit.committer + "\n";
content += "\n";
content += commit.message;
return this.writeObject("commit", this.stringToArray(content));
}
writeCommit(commit) {
return this.writeTreeRecursive(commit.tree)
.then(treeId => {
commit.tree = treeId;
return this.writePlainCommit(commit);
});
}
// Refs commands
getRef(ref) {
return this.readFile(ref)
......@@ -565,4 +686,151 @@ class Git {
return Promise.reject("No HEAD ref");
});
}
// Pair-repo actions
importObject(other, id) {
return this.readObject(id)
.then(object => {
// Already exists
return false;
}, () => {
let object;
return other.readObject(id)
.then(o => {
object = o;
return this.writeObject(object.type, object.content);
})
.then(newId => {
if(newId != id) {
return Promise.reject("SHA1 mismatch during importing " + id + " (received " + newId + ") from " + other + " to " + this);
}
return true;
});
});
}
importObjectWithDependencies(other, id) {
return this.importObject(other, id)
.then(imported => {
if(!imported) {
// Already have object and dependencies locally
return;
}
return this.readUnknownObject(id)
.then(object => {
if(object.type == "blob") {
return this.importBlobDependencies(other, object);
} else if(object.type == "tree") {
return this.importTreeDependencies(other, object);
} else if(object.type == "commit") {
return this.importCommitDependencies(other, object);
}
});
});
}
importBlobDependencies(other, blob) {
// Blob has no dependencies
return Promise.resolve();
}
importTreeDependencies(other, tree) {
return Promise.all(
tree.content.map(item => {
return this.importObjectWithDependencies(other, item.id);
})
);
}
importCommitDependencies(other, commit) {
return Promise.all(
[
this.importObjectWithDependencies(other, commit.content.tree), // Import tree
].concat(
commit.content.parents.map(parent => {
return this.importObjectWithDependencies(other, parent) // Import parents
})
)
);
}
makeTreeDelta(base, changes) {
// changes:
// [
// {
// name: "dir",
// type: "tree",
// content: [
// {
// name: "subdir",
// type: "tree",
// content: [
// {
// name: "subfile",
// type: "blob",
// content: "neworchangedfile"
// }
// ]
// }
// ]
// },
// {
// name: "file",
// remove: true
// },
// {
// name: "olddir",
// type: "blob",
// content: "fds"
// }
// ]
//
// Should be read as:
// 1. Set <dir/subdir/subfile> blob to "neworchangedfile"
// 2. Remove <file>
// 3. Remove tree <olddir> and add blob <olddir>
let promise = Promise.all(
changes.map(change => {
let treeItemIndex = base.findIndex(item => item.name == change.name);
if(change.remove) {
// Remove
if(treeItemIndex > -1) {
base.splice(treeItemIndex, 1);
}
} else if(change.type == "blob") {
if(treeItemIndex == -1) {
// Add blob
base.push(change);
} else {
// Change type to blob or change blob
base[treeItemIndex] = change;
}
} else if(change.type == "tree") {
if(treeItemIndex == -1) {
// Add tree
base.push(change);
} else if(tree[treeItemIndex].type != "tree") {
// Change type to tree
base[treeItemIndex] = change;
} else {
// Change tree
let id = base[treeItemIndex].id;
delete base[treeItemIndex].id;
return this.readUnknownObject(id)
.then(subTree => {
return this.makeTreeDelta(subTree.content, change.content);
});
}
}
return Promise.resolve();
})
);
return promise.then(() => base);
}
toString() {
return "<Git " + this.root + ">";
}
};
\ No newline at end of file
......@@ -22,6 +22,64 @@ Use `getBranchCommit("master")` to get commit the branch references or use `read
Use `getHead()`.
### I can read objects, can I write them?
Yes. Currently ZeroGit only supports loose objects writing. Use `writeObject(type, plainContent)` which hashes object and returns its new SHA.
### What about auto formatting saved objects?
You can use:
1. `writeBlob(rawBlobContent)`
2. `writeTree(items)` where `items` have the same format as `readUnknownObject()` result for trees.
3. While `writeTree(items)` expects each subtree to be `{type: "tree", name: "mydir", id: "0123456789abcdefghij"}`, `writeTreeRecursive` also supports `{type: "tree", name: "mydir", content: [{type: "blob", name: "subfile", content: "myblob"}]}`. So you can build subtrees without need to call `writeTree` several times.
4. `writePlainCommit(commit)` expects `commit` to have the same format as the result of `readUnknownObject()` for commits, so it makes you set `tree` as SHA.
5. `writeCommit(commit)` works like `writePlainCommit(commit)` but also builds tree with `writeTreeRecursive(commit.tree)` and sets new SHA as commit tree.
### Can I simplify committing?
`makeTreeDelta` gives you power not to recreate all tree. It accepts `base` as result of `readUnknownObject().content` and `changes` as array of items where each item can be either `{name: "...", type: "tree", content: [...]}` or `{name: "...", remove: true}` or `{name: "...", type: "blob", content: "..."]`. The resulting value is new tree. Note that `base` argument is changed.
For example,
[
{
name: "dir",
type: "tree",
content: [
{
name: "subdir",
type: "tree",
content: [
{
name: "subfile",
type: "blob",
content: "neworchangedfile"
}
]
}
]
},
{
name: "file",
remove: true
},
{
name: "olddir",
type: "blob",
content: "fds"
}
]
...should be read as:
1. Set `dir/subdir/subfile` blob to `neworchangedfile`
2. Remove `file`
3. Replace tree `olddir` with blob `olddir`
4. Leave all not mentioned files (if there was `dir/somefile` or `readme.md` they would be added to resulting tree).
### And some actions between repositories?
Yes! Run `importObject(otherGit, sha)` to read `sha` from `otherGit` and save it to `this`. And `importObjectWithDependencies(otherGit, sha)` also imports commit parents and tree, tree subitems, etc.
## Getting loose objects
For getting loose object, ZeroGit splits SHA to 2-and-18 parts, reads file `objects/01/23456789abcdefghij` and deflates it. The result is:
......
/*
A JavaScript implementation of the SHA family of hashes, as
defined in FIPS PUB 180-4 and FIPS PUB 202, as well as the corresponding
HMAC implementation as defined in FIPS PUB 198a
Copyright Brian Turek 2008-2017
Distributed under the BSD License
See http://caligatio.github.com/jsSHA/ for more information
Several functions taken from Paul Johnston
*/
'use strict';(function(G){function r(d,b,c){var h=0,a=[],f=0,g,m,k,e,l,p,q,t,w=!1,n=[],u=[],v,r=!1;c=c||{};g=c.encoding||"UTF8";v=c.numRounds||1;if(v!==parseInt(v,10)||1>v)throw Error("numRounds must a integer >= 1");if("SHA-1"===d)l=512,p=z,q=H,e=160,t=function(a){return a.slice()};else throw Error("Chosen SHA variant is not supported");k=A(b,g);m=x(d);this.setHMACKey=function(a,f,b){var c;if(!0===w)throw Error("HMAC key already set");if(!0===r)throw Error("Cannot set HMAC key after calling update");
g=(b||{}).encoding||"UTF8";f=A(f,g)(a);a=f.binLen;f=f.value;c=l>>>3;b=c/4-1;if(c<a/8){for(f=q(f,a,0,x(d),e);f.length<=b;)f.push(0);f[b]&=4294967040}else if(c>a/8){for(;f.length<=b;)f.push(0);f[b]&=4294967040}for(a=0;a<=b;a+=1)n[a]=f[a]^909522486,u[a]=f[a]^1549556828;m=p(n,m);h=l;w=!0};this.update=function(b){var e,g,c,d=0,q=l>>>5;e=k(b,a,f);b=e.binLen;g=e.value;e=b>>>5;for(c=0;c<e;c+=q)d+l<=b&&(m=p(g.slice(c,c+q),m),d+=l);h+=d;a=g.slice(d>>>5);f=b%l;r=!0};this.getHash=function(b,g){var c,k,l,p;if(!0===
w)throw Error("Cannot call getHash after setting HMAC key");l=B(g);switch(b){case "HEX":c=function(a){return C(a,e,l)};break;case "B64":c=function(a){return D(a,e,l)};break;case "BYTES":c=function(a){return E(a,e)};break;case "ARRAYBUFFER":try{k=new ArrayBuffer(0)}catch(I){throw Error("ARRAYBUFFER not supported by this environment");}c=function(a){return F(a,e)};break;default:throw Error("format must be HEX, B64, BYTES, or ARRAYBUFFER");}p=q(a.slice(),f,h,t(m),e);for(k=1;k<v;k+=1)p=q(p,e,0,x(d),e);
return c(p)};this.getHMAC=function(b,g){var c,k,n,r;if(!1===w)throw Error("Cannot call getHMAC without first setting HMAC key");n=B(g);switch(b){case "HEX":c=function(a){return C(a,e,n)};break;case "B64":c=function(a){return D(a,e,n)};break;case "BYTES":c=function(a){return E(a,e)};break;case "ARRAYBUFFER":try{c=new ArrayBuffer(0)}catch(I){throw Error("ARRAYBUFFER not supported by this environment");}c=function(a){return F(a,e)};break;default:throw Error("outputFormat must be HEX, B64, BYTES, or ARRAYBUFFER");
}k=q(a.slice(),f,h,t(m),e);r=p(u,x(d));r=q(k,e,l,r,e);return c(r)}}function C(d,b,c){var h="";b/=8;var a,f;for(a=0;a<b;a+=1)f=d[a>>>2]>>>8*(3+a%4*-1),h+="0123456789abcdef".charAt(f>>>4&15)+"0123456789abcdef".charAt(f&15);return c.outputUpper?h.toUpperCase():h}function D(d,b,c){var h="",a=b/8,f,g,m;for(f=0;f<a;f+=3)for(g=f+1<a?d[f+1>>>2]:0,m=f+2<a?d[f+2>>>2]:0,m=(d[f>>>2]>>>8*(3+f%4*-1)&255)<<16|(g>>>8*(3+(f+1)%4*-1)&255)<<8|m>>>8*(3+(f+2)%4*-1)&255,g=0;4>g;g+=1)8*f+6*g<=b?h+="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".charAt(m>>>
6*(3-g)&63):h+=c.b64Pad;return h}function E(d,b){var c="",h=b/8,a,f;for(a=0;a<h;a+=1)f=d[a>>>2]>>>8*(3+a%4*-1)&255,c+=String.fromCharCode(f);return c}function F(d,b){var c=b/8,h,a=new ArrayBuffer(c),f;f=new Uint8Array(a);for(h=0;h<c;h+=1)f[h]=d[h>>>2]>>>8*(3+h%4*-1)&255;return a}function B(d){var b={outputUpper:!1,b64Pad:"=",shakeLen:-1};d=d||{};b.outputUpper=d.outputUpper||!1;!0===d.hasOwnProperty("b64Pad")&&(b.b64Pad=d.b64Pad);if("boolean"!==typeof b.outputUpper)throw Error("Invalid outputUpper formatting option");
if("string"!==typeof b.b64Pad)throw Error("Invalid b64Pad formatting option");return b}function A(d,b){var c;switch(b){case "UTF8":case "UTF16BE":case "UTF16LE":break;default:throw Error("encoding must be UTF8, UTF16BE, or UTF16LE");}switch(d){case "HEX":c=function(b,a,f){var g=b.length,c,d,e,l,p;if(0!==g%2)throw Error("String of HEX type must be in byte increments");a=a||[0];f=f||0;p=f>>>3;for(c=0;c<g;c+=2){d=parseInt(b.substr(c,2),16);if(isNaN(d))throw Error("String of HEX type contains invalid characters");
l=(c>>>1)+p;for(e=l>>>2;a.length<=e;)a.push(0);a[e]|=d<<8*(3+l%4*-1)}return{value:a,binLen:4*g+f}};break;case "TEXT":c=function(c,a,f){var g,d,k=0,e,l,p,q,t,n;a=a||[0];f=f||0;p=f>>>3;if("UTF8"===b)for(n=3,e=0;e<c.length;e+=1)for(g=c.charCodeAt(e),d=[],128>g?d.push(g):2048>g?(d.push(192|g>>>6),d.push(128|g&63)):55296>g||57344<=g?d.push(224|g>>>12,128|g>>>6&63,128|g&63):(e+=1,g=65536+((g&1023)<<10|c.charCodeAt(e)&1023),d.push(240|g>>>18,128|g>>>12&63,128|g>>>6&63,128|g&63)),l=0;l<d.length;l+=1){t=k+
p;for(q=t>>>2;a.length<=q;)a.push(0);a[q]|=d[l]<<8*(n+t%4*-1);k+=1}else if("UTF16BE"===b||"UTF16LE"===b)for(n=2,d="UTF16LE"===b&&!0||"UTF16LE"!==b&&!1,e=0;e<c.length;e+=1){g=c.charCodeAt(e);!0===d&&(l=g&255,g=l<<8|g>>>8);t=k+p;for(q=t>>>2;a.length<=q;)a.push(0);a[q]|=g<<8*(n+t%4*-1);k+=2}return{value:a,binLen:8*k+f}};break;case "B64":c=function(b,a,f){var c=0,d,k,e,l,p,q,n;if(-1===b.search(/^[a-zA-Z0-9=+\/]+$/))throw Error("Invalid character in base-64 string");k=b.indexOf("=");b=b.replace(/\=/g,
"");if(-1!==k&&k<b.length)throw Error("Invalid '=' found in base-64 string");a=a||[0];f=f||0;q=f>>>3;for(k=0;k<b.length;k+=4){p=b.substr(k,4);for(e=l=0;e<p.length;e+=1)d="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".indexOf(p[e]),l|=d<<18-6*e;for(e=0;e<p.length-1;e+=1){n=c+q;for(d=n>>>2;a.length<=d;)a.push(0);a[d]|=(l>>>16-8*e&255)<<8*(3+n%4*-1);c+=1}}return{value:a,binLen:8*c+f}};break;case "BYTES":c=function(b,a,c){var d,m,k,e,l;a=a||[0];c=c||0;k=c>>>3;for(m=0;m<b.length;m+=
1)d=b.charCodeAt(m),l=m+k,e=l>>>2,a.length<=e&&a.push(0),a[e]|=d<<8*(3+l%4*-1);return{value:a,binLen:8*b.length+c}};break;case "ARRAYBUFFER":try{c=new ArrayBuffer(0)}catch(h){throw Error("ARRAYBUFFER not supported by this environment");}c=function(b,a,c){var d,m,k,e,l;a=a||[0];c=c||0;m=c>>>3;l=new Uint8Array(b);for(d=0;d<b.byteLength;d+=1)e=d+m,k=e>>>2,a.length<=k&&a.push(0),a[k]|=l[d]<<8*(3+e%4*-1);return{value:a,binLen:8*b.byteLength+c}};break;default:throw Error("format must be HEX, TEXT, B64, BYTES, or ARRAYBUFFER");
}return c}function n(d,b){return d<<b|d>>>32-b}function u(d,b){var c=(d&65535)+(b&65535);return((d>>>16)+(b>>>16)+(c>>>16)&65535)<<16|c&65535}function y(d,b,c,h,a){var f=(d&65535)+(b&65535)+(c&65535)+(h&65535)+(a&65535);return((d>>>16)+(b>>>16)+(c>>>16)+(h>>>16)+(a>>>16)+(f>>>16)&65535)<<16|f&65535}function x(d){var b=[];if("SHA-1"===d)b=[1732584193,4023233417,2562383102,271733878,3285377520];else throw Error("No SHA variants supported");return b}function z(d,b){var c=[],h,a,f,g,m,k,e;h=b[0];a=b[1];
f=b[2];g=b[3];m=b[4];for(e=0;80>e;e+=1)c[e]=16>e?d[e]:n(c[e-3]^c[e-8]^c[e-14]^c[e-16],1),k=20>e?y(n(h,5),a&f^~a&g,m,1518500249,c[e]):40>e?y(n(h,5),a^f^g,m,1859775393,c[e]):60>e?y(n(h,5),a&f^a&g^f&g,m,2400959708,c[e]):y(n(h,5),a^f^g,m,3395469782,c[e]),m=g,g=f,f=n(a,30),a=h,h=k;b[0]=u(h,b[0]);b[1]=u(a,b[1]);b[2]=u(f,b[2]);b[3]=u(g,b[3]);b[4]=u(m,b[4]);return b}function H(d,b,c,h){var a;for(a=(b+65>>>9<<4)+15;d.length<=a;)d.push(0);d[b>>>5]|=128<<24-b%32;b+=c;d[a]=b&4294967295;d[a-1]=b/4294967296|0;
b=d.length;for(a=0;a<b;a+=16)h=z(d.slice(a,a+16),h);return h}"function"===typeof define&&define.amd?define(function(){return r}):"undefined"!==typeof exports?("undefined"!==typeof module&&module.exports&&(module.exports=r),exports=r):G.jsSHA=r})(this);
......@@ -85,6 +85,7 @@
<script type="text/javascript" src="../../js/ZeroAuth.js"></script>
<script type="text/javascript" src="../../js/ZeroDB.js"></script>
<script type="text/javascript" src="../../js/pako.js"></script>
<script type="text/javascript" src="../../js/sha.js"></script>
<script type="text/javascript" src="../../js/git.js"></script>
<script type="text/javascript" src="../../js/repo.js"></script>
<script type="text/javascript" src="../js/path_common.js"></script>
......
......@@ -46,6 +46,8 @@
<script type="text/javascript" src="../../js/ZeroFS.js"></script>
<script type="text/javascript" src="../../js/ZeroAuth.js"></script>
<script type="text/javascript" src="../../js/ZeroDB.js"></script>
<script type="text/javascript" src="../../js/pako.js"></script>
<script type="text/javascript" src="../../js/sha.js"></script>
<script type="text/javascript" src="../../js/git.js"></script>
<script type="text/javascript" src="../../js/repo.js"></script>
<script type="text/javascript" src="../js/user_common.js"></script>
......
......@@ -85,6 +85,7 @@
<script type="text/javascript" src="../js/ZeroAuth.js"></script>
<script type="text/javascript" src="../js/ZeroDB.js"></script>
<script type="text/javascript" src="../js/pako.js"></script>
<script type="text/javascript" src="../js/sha.js"></script>
<script type="text/javascript" src="../js/git.js"></script>
<script type="text/javascript" src="../js/repo.js"></script>
<script type="text/javascript" src="js/path_common.js"></script>
......
......@@ -75,6 +75,8 @@
<script type="text/javascript" src="../../js/ZeroFS.js"></script>
<script type="text/javascript" src="../../js/ZeroAuth.js"></script>
<script type="text/javascript" src="../../js/ZeroDB.js"></script>
<script type="text/javascript" src="../../js/pako.js"></script>
<script type="text/javascript" src="../../js/sha.js"></script>
<script type="text/javascript" src="../../js/git.js"></script>
<script type="text/javascript" src="../../js/repo.js"></script>
<script type="text/javascript" src="../js/common.js"></script>
......
......@@ -71,6 +71,8 @@
<script type="text/javascript" src="../../../js/ZeroFS.js"></script>
<script type="text/javascript" src="../../../js/ZeroAuth.js"></script>
<script type="text/javascript" src="../../../js/ZeroDB.js"></script>
<script type="text/javascript" src="../../../js/pako.js"></script>
<script type="text/javascript" src="../../../js/sha.js"></script>
<script type="text/javascript" src="../../../js/git.js"></script>
<script type="text/javascript" src="../../../js/repo.js"></script>
<script type="text/javascript" src="../../js/common.js"></script>
......
......@@ -87,6 +87,8 @@
<script type="text/javascript" src="../../../js/ZeroFS.js"></script>
<script type="text/javascript" src="../../../js/ZeroAuth.js"></script>
<script type="text/javascript" src="../../../js/ZeroDB.js"></script>
<script type="text/javascript" src="../../../js/pako.js"></script>
<script type="text/javascript" src="../../../js/sha.js"></script>
<script type="text/javascript" src="../../../js/git.js"></script>
<script type="text/javascript" src="../../../js/repo.js"></script>
<script type="text/javascript" src="../../js/common.js"></script>
......
......@@ -70,6 +70,8 @@
<script type="text/javascript" src="../../js/ZeroFS.js"></script>
<script type="text/javascript" src="../../js/ZeroAuth.js"></script>
<script type="text/javascript" src="../../js/ZeroDB.js"></script>
<script type="text/javascript" src="../../js/pako.js"></script>
<script type="text/javascript" src="../../js/sha.js"></script>
<script type="text/javascript" src="../../js/git.js"></script>
<script type="text/javascript" src="../../js/repo.js"></script>
<script type="text/javascript" src="../js/common.js"></script>
......
......@@ -112,6 +112,8 @@
<script type="text/javascript" src="../../js/ZeroFS.js"></script>
<script type="text/javascript" src="../../js/ZeroAuth.js"></script>
<script type="text/javascript" src="../../js/ZeroDB.js"></script>
<script type="text/javascript" src="../../js/pako.js"></script>
<script type="text/javascript" src="../../js/sha.js"></script>
<script type="text/javascript" src="../../js/git.js"></script>
<script type="text/javascript" src="../../js/repo.js"></script>
<script type="text/javascript" src="../js/common.js"></script>
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment