diff --git a/cmd/magneticow/api.go b/cmd/magneticow/api.go
index 69fd709..fb96634 100644
--- a/cmd/magneticow/api.go
+++ b/cmd/magneticow/api.go
@@ -13,7 +13,7 @@ import (
"go.uber.org/zap"
)
-func apiTorrentsHandler(w http.ResponseWriter, r *http.Request) {
+func apiTorrents(w http.ResponseWriter, r *http.Request) {
// @lastOrderedValue AND @lastID are either both supplied or neither of them should be supplied
// at all; and if that is NOT the case, then return an error.
if q := r.URL.Query(); !((q.Get("lastOrderedValue") != "" && q.Get("lastID") != "") ||
@@ -29,6 +29,7 @@ func apiTorrentsHandler(w http.ResponseWriter, r *http.Request) {
Ascending *bool `schema:"ascending"`
LastOrderedValue *float64 `schema:"lastOrderedValue"`
LastID *uint64 `schema:"lastID"`
+ Limit *uint `schema:"limit"`
}
if err := decoder.Decode(&tq, r.URL.Query()); err != nil {
respondError(w, 400, "error while parsing the URL: %s", err.Error())
@@ -69,27 +70,26 @@ func apiTorrentsHandler(w http.ResponseWriter, r *http.Request) {
}
}
+ if tq.Limit == nil {
+ tq.Limit = new(uint)
+ *tq.Limit = 20
+ }
+
torrents, err := database.QueryTorrents(
*tq.Query, *tq.Epoch, orderBy,
- *tq.Ascending, N_TORRENTS, tq.LastOrderedValue, tq.LastID)
+ *tq.Ascending, *tq.Limit, tq.LastOrderedValue, tq.LastID)
if err != nil {
respondError(w, 400, "query error: %s", err.Error())
return
}
- // TODO: use plain Marshal
- jm, err := json.MarshalIndent(torrents, "", " ")
- if err != nil {
- respondError(w, 500, "json marshalling error: %s", err.Error())
- return
- }
-
- if _, err = w.Write(jm); err != nil {
- zap.L().Warn("couldn't write http.ResponseWriter", zap.Error(err))
+ w.Header().Set("Content-Type", "application/json; charset=utf-8")
+ if err = json.NewEncoder(w).Encode(torrents); err != nil {
+ zap.L().Warn("JSON encode error", zap.Error(err))
}
}
-func apiTorrentsInfohashHandler(w http.ResponseWriter, r *http.Request) {
+func apiTorrent(w http.ResponseWriter, r *http.Request) {
infohashHex := mux.Vars(r)["infohash"]
infohash, err := hex.DecodeString(infohashHex)
@@ -107,19 +107,13 @@ func apiTorrentsInfohashHandler(w http.ResponseWriter, r *http.Request) {
return
}
- // TODO: use plain Marshal
- jm, err := json.MarshalIndent(torrent, "", " ")
- if err != nil {
- respondError(w, 500, "json marshalling error: %s", err.Error())
- return
- }
-
- if _, err = w.Write(jm); err != nil {
- zap.L().Warn("couldn't write http.ResponseWriter", zap.Error(err))
+ w.Header().Set("Content-Type", "application/json; charset=utf-8")
+ if err = json.NewEncoder(w).Encode(torrent); err != nil {
+ zap.L().Warn("JSON encode error", zap.Error(err))
}
}
-func apiFilesInfohashHandler(w http.ResponseWriter, r *http.Request) {
+func apiFilelist(w http.ResponseWriter, r *http.Request) {
infohashHex := mux.Vars(r)["infohash"]
infohash, err := hex.DecodeString(infohashHex)
@@ -137,19 +131,13 @@ func apiFilesInfohashHandler(w http.ResponseWriter, r *http.Request) {
return
}
- // TODO: use plain Marshal
- jm, err := json.MarshalIndent(files, "", " ")
- if err != nil {
- respondError(w, 500, "json marshalling error: %s", err.Error())
- return
- }
-
- if _, err = w.Write(jm); err != nil {
- zap.L().Warn("couldn't write http.ResponseWriter", zap.Error(err))
+ w.Header().Set("Content-Type", "application/json; charset=utf-8")
+ if err = json.NewEncoder(w).Encode(files); err != nil {
+ zap.L().Warn("JSON encode error", zap.Error(err))
}
}
-func apiStatisticsHandler(w http.ResponseWriter, r *http.Request) {
+func apiStatistics(w http.ResponseWriter, r *http.Request) {
from := r.URL.Query().Get("from")
// TODO: use gorilla?
@@ -175,15 +163,9 @@ func apiStatisticsHandler(w http.ResponseWriter, r *http.Request) {
return
}
- // TODO: use plain Marshal
- jm, err := json.MarshalIndent(stats, "", " ")
- if err != nil {
- respondError(w, 500, "json marshalling error: %s", err.Error())
- return
- }
-
- if _, err = w.Write(jm); err != nil {
- zap.L().Warn("couldn't write http.ResponseWriter", zap.Error(err))
+ w.Header().Set("Content-Type", "application/json; charset=utf-8")
+ if err = json.NewEncoder(w).Encode(stats); err != nil {
+ zap.L().Warn("JSON encode error", zap.Error(err))
}
}
diff --git a/cmd/magneticow/data/static/scripts/common.js b/cmd/magneticow/data/static/scripts/common.js
index f935705..7c7a0bf 100644
--- a/cmd/magneticow/data/static/scripts/common.js
+++ b/cmd/magneticow/data/static/scripts/common.js
@@ -24,6 +24,14 @@ function fileSize(fileSizeInBytes) {
return Math.max(fileSizeInBytes, 0.1).toFixed(1) + byteUnits[i];
}
+function humaniseDate(unixTime) {
+ return (new Date(unixTime * 1000)).toLocaleDateString("en-GB", {
+ day: "2-digit",
+ month: "2-digit",
+ year: "numeric"
+ });
+}
+
/**
* Returns the ISO 8601 week number for this date.
*
diff --git a/cmd/magneticow/data/static/scripts/torrent.js b/cmd/magneticow/data/static/scripts/torrent.js
index 1c9cbc9..4be0c8d 100644
--- a/cmd/magneticow/data/static/scripts/torrent.js
+++ b/cmd/magneticow/data/static/scripts/torrent.js
@@ -9,52 +9,37 @@
window.onload = function() {
- var pre_element = document.getElementsByTagName("pre")[0];
- var paths = pre_element.textContent.replace(/\s+$/, "").split("\n");
- paths.sort(naturalSort);
- paths = paths.map(function(path) { return path.split('/'); });
- pre_element.textContent = stringify(structurise(paths)).join("\n");
-};
+ let infoHash = window.location.pathname.split("/")[2];
+ fetch("/api/v0.1/torrents/" + infoHash).then(x => x.json()).then(x => {
+ document.querySelector("title").innerText = x.name + " - magneticow";
-function structurise(paths) {
- var items = [];
- for(var i = 0, l = paths.length; i < l; i++) {
- var path = paths[i];
- var name = path[0];
- var rest = path.slice(1);
- var item = null;
- for(var j = 0, m = items.length; j < m; j++) {
- if(items[j].name === name) {
- item = items[j];
- break;
+ const template = document.getElementById("main-template").innerHTML;
+ document.querySelector("main").innerHTML = Mustache.render(template, {
+ name: x.name,
+ infoHash: x.infoHash,
+ sizeHumanised: fileSize(x.size),
+ discoveredOnHumanised: humaniseDate(x.discoveredOn),
+ nFiles: x.nFiles,
+ });
+
+ fetch("/api/v0.1/torrents/" + infoHash + "/filelist").then(x => x.json()).then(x => {
+ const tree = new VanillaTree('#fileTree', {
+ placeholder: 'Loading...',
+ });
+
+ for (let e of x) {
+ let pathElems = e.path.split("/");
+
+ for (let i = 0; i < pathElems.length; i++) {
+ tree.add({
+ id: pathElems.slice(0, i + 1).join("/"),
+ parent: i >= 1 ? pathElems.slice(0, i).join("/") : undefined,
+ label: pathElems[i] + ( i === pathElems.length - 1 ? " " + fileSize(e.size) + "" : ""),
+ opened: true,
+ });
+ }
}
- }
- if(item === null) {
- item = {name: name, children: []};
- items.push(item);
- }
- if(rest.length > 0) {
- item.children.push(rest);
- }
- }
- for(i = 0, l = items.length; i < l; i++) {
- item = items[i];
- item.children = structurise(item.children);
- }
- return items;
-}
-
-
-function stringify(items) {
- var lines = [];
- for(var i = 0, l = items.length; i < l; i++) {
- var item = items[i];
- lines.push(item.name);
- var subLines = stringify(item.children);
- for(var j = 0, m = subLines.length; j < m; j++) {
- lines.push(" " + subLines[j]);
- }
- }
- return lines;
-}
+ });
+ });
+};
diff --git a/cmd/magneticow/data/static/scripts/torrents.js b/cmd/magneticow/data/static/scripts/torrents.js
index 8ab5589..44f0bc8 100644
--- a/cmd/magneticow/data/static/scripts/torrents.js
+++ b/cmd/magneticow/data/static/scripts/torrents.js
@@ -104,11 +104,7 @@ function load() {
for (let t of torrents) {
t.size = fileSize(t.size);
- t.discoveredOn = (new Date(t.discoveredOn * 1000)).toLocaleDateString("en-GB", {
- day: "2-digit",
- month: "2-digit",
- year: "numeric"
- });
+ t.discoveredOn = humaniseDate(t.discoveredOn);
ul.innerHTML += Mustache.render(template, t);
}
diff --git a/cmd/magneticow/data/static/scripts/vanillatree-v0.0.3.js b/cmd/magneticow/data/static/scripts/vanillatree-v0.0.3.js
new file mode 100644
index 0000000..202ae07
--- /dev/null
+++ b/cmd/magneticow/data/static/scripts/vanillatree-v0.0.3.js
@@ -0,0 +1,214 @@
+(function (root, factory) {
+ if (typeof define == 'function' && define.amd) {
+ define( factory );
+ } else if (typeof module === 'object' && module.exports) {
+ module.exports = factory();
+ } else {
+ root.VanillaTree = factory();
+ }
+}(this, function () {
+ "use strict";
+ // Look at the Balalaika https://github.com/finom/balalaika
+ var $=function(n,e,k,h,p,m,l,b,d,g,f,c){c=function(a,b){return new c.i(a,b)};c.i=function(a,d){k.push.apply(this,a?a.nodeType||a==n?[a]:""+a===a?/' + this.placeholder + ''
+ } else if( p = this.tree.querySelector( '.vtree-placeholder' ) ) {
+ this.tree.removeChild( p );
+ }
+ return this;
+ },
+ getLeaf: function( id, notThrow ) {
+ var leaf = $( '[data-vtree-id="' + id + '"]', this.tree )[ 0 ];
+ if( !notThrow && !leaf ) throw Error( 'No VanillaTree leaf with id "' + id + '"' )
+ return leaf;
+ },
+ getChildList: function( id ) {
+ var list,
+ parent;
+ if( id ) {
+ parent = this.getLeaf( id );
+ if( !( list = $( 'ul', parent )[ 0 ] ) ) {
+ list = parent.appendChild( create( 'ul', {
+ className: 'vtree-subtree'
+ }) );
+ }
+ } else {
+ list = this.tree;
+ }
+
+ return list;
+ },
+ add: function( options ) {
+ // see https://github.com/finom/vanillatree/issues/8
+ if(this.getLeaf( options.id,true )){return;}// don't add leaves that already exist
+ var id,
+ leaf = create( 'li', {
+ className: 'vtree-leaf'
+ }),
+ parentList = this.getChildList( options.parent );
+
+ leaf.setAttribute( 'data-vtree-id', id = options.id || Math.random() );
+
+ leaf.appendChild( create( 'span', {
+ className: 'vtree-toggle'
+ }) );
+
+ leaf.appendChild( create( 'a', {
+ className: 'vtree-leaf-label',
+ innerHTML: options.label
+ }) );
+
+ parentList.appendChild( leaf );
+
+ if( parentList !== this.tree ) {
+ parentList.parentNode.classList.add( 'vtree-has-children' );
+ }
+
+ this.leafs[ id ] = options;
+
+ if( !options.opened ) {
+ this.close( id );
+ }
+
+ if( options.selected ) {
+ this.select( id );
+ }
+
+ return this._placeholder()._dispatch( 'add', id );
+ },
+ move: function( id, parentId ) {
+ var leaf = this.getLeaf( id ),
+ oldParent = leaf.parentNode,
+ newParent = this.getLeaf( parentId, true );
+
+ if( newParent ) {
+ newParent.classList.add( 'vtree-has-children' );
+ }
+
+ this.getChildList( parentId ).appendChild( leaf );
+ oldParent.parentNode.classList.toggle( 'vtree-has-children', !!oldParent.children.length );
+
+ return this._dispatch( 'move', id );
+ },
+ remove: function( id ) {
+ var leaf = this.getLeaf( id ),
+ oldParent = leaf.parentNode;
+ oldParent.removeChild( leaf );
+ oldParent.parentNode.classList.toggle( 'vtree-has-children', !!oldParent.children.length );
+
+ return this._placeholder()._dispatch( 'remove', id );
+ },
+ open: function( id ) {
+ this.getLeaf( id ).classList.remove( 'closed' );
+ return this._dispatch( 'open', id );
+ },
+ close: function( id ) {
+ this.getLeaf( id ).classList.add( 'closed' );
+ return this._dispatch( 'close', id );
+ },
+ toggle: function( id ) {
+ return this[ this.getLeaf( id ).classList.contains( 'closed' ) ? 'open' : 'close' ]( id );
+ },
+ select: function( id ) {
+ var leaf = this.getLeaf( id );
+
+ if( !leaf.classList.contains( 'vtree-selected' ) ) {
+ $( 'li.vtree-leaf', this.tree ).forEach( function( leaf ) {
+ leaf.classList.remove( 'vtree-selected' );
+ });
+
+ leaf.classList.add( 'vtree-selected' );
+ this._dispatch( 'select', id );
+ }
+
+ return this;
+ }
+ };
+
+ return Tree;
+ // Look at the Balalaika https://github.com/finom/balalaika
+}));
diff --git a/cmd/magneticow/data/static/styles/torrent.css b/cmd/magneticow/data/static/styles/torrent.css
index 6527a40..610ac6c 100644
--- a/cmd/magneticow/data/static/styles/torrent.css
+++ b/cmd/magneticow/data/static/styles/torrent.css
@@ -37,7 +37,8 @@ header > div {
}
#title h2 {
- margin-bottom: 0px;
+ margin-bottom: 0;
+ word-break: break-all;
}
#title {
@@ -49,12 +50,6 @@ header > div {
color: inherit;
}
-pre {
- background-color: black;
- color: white;
- overflow: auto;
-}
-
table {
max-width: 700px;
width: 700px;
@@ -77,4 +72,9 @@ th {
text-align: left;
width: 1%;
+}
+
+tt {
+ font-style: italic;
+ font-size: smaller;
}
\ No newline at end of file
diff --git a/cmd/magneticow/data/static/styles/torrents.css b/cmd/magneticow/data/static/styles/torrents.css
index 5485068..e0e2c86 100644
--- a/cmd/magneticow/data/static/styles/torrents.css
+++ b/cmd/magneticow/data/static/styles/torrents.css
@@ -79,6 +79,7 @@ ul li div {
ul li div h3 {
margin: 0;
+ word-break: break-all;
}
a {
diff --git a/cmd/magneticow/data/static/styles/vanillatree-v0.0.3.css b/cmd/magneticow/data/static/styles/vanillatree-v0.0.3.css
new file mode 100644
index 0000000..683c124
--- /dev/null
+++ b/cmd/magneticow/data/static/styles/vanillatree-v0.0.3.css
@@ -0,0 +1,96 @@
+.vtree ul.vtree-subtree, .vtree li.vtree-leaf {
+ margin: 0;
+ padding: 0;
+ list-style-type: none;
+ position: relative;
+}
+
+.vtree li.vtree-leaf {
+ background-position: -90px 0;
+ background-repeat: repeat-y;
+ min-height: 18px;
+ line-height: 18px;
+}
+
+.vtree li.vtree-leaf::before {
+ content: '';
+ width: 18px;
+ height: 18px;
+ position: absolute;
+ background-position: -36px 0;
+}
+
+.vtree li.vtree-leaf li.vtree-leaf {
+ margin-left: 18px;
+}
+
+.vtree li.vtree-leaf:last-child {
+ background-image: none;
+}
+
+.vtree li.vtree-leaf.closed ul.vtree-subtree {
+ display: none;
+}
+
+.vtree li.vtree-leaf.vtree-has-children > span.vtree-toggle {
+ display: block;
+ width: 18px;
+ height: 18px;
+ background-position: -72px 0;
+ position: absolute;
+ left: 0;
+ top: 0;
+}
+
+.vtree li.vtree-leaf.vtree-has-children.closed > span.vtree-toggle {
+ background-position: -54px 0;
+}
+
+.vtree a.vtree-leaf-label {
+ line-height: 18px;
+ display: inline-block;
+ vertical-align: top;
+ cursor: pointer;
+ max-width: 100%;
+ margin-left: 18px;
+ padding: 0 2px;
+}
+
+.vtree li.vtree-leaf a.vtree-leaf-label:hover {
+ background-color: #e7f4f9;
+ outline: 1px solid #d8f0fa;
+}
+
+.vtree li.vtree-leaf.vtree-selected > a.vtree-leaf-label {
+ background-color: #beebff;
+ outline: 1px solid #99defd;
+}
+
+.vtree-contextmenu {
+ position: absolute;
+ z-index: 9999999;
+ border: solid 1px #ccc;
+ background: #eee;
+ padding: 0px;
+ margin: 0px;
+ display: none;
+}
+
+.vtree-contextmenu li {
+ list-style: none;
+ padding: 1px 5px;
+ margin: 0px;
+ color: #333;
+ line-height: 20px;
+ height: 20px;
+ cursor: default;
+}
+
+.vtree-contextmenu li:hover {
+ color: #fff;
+ background-color: #3399ff;
+}
+
+.vtree li.vtree-leaf, .vtree li.vtree-leaf::before, .vtree li.vtree-leaf.vtree-has-children > span.vtree-toggle {
+ background-image: url();
+}
diff --git a/cmd/magneticow/data/templates/torrent.html b/cmd/magneticow/data/templates/torrent.html
index e0af377..102a2ee 100644
--- a/cmd/magneticow/data/templates/torrent.html
+++ b/cmd/magneticow/data/templates/torrent.html
@@ -3,12 +3,48 @@
- {{ .T.Name }} - magnetico
+ Loading... - magneticow
+
+
-
-
+
+
+
+
+
+
+
+
+
-
-
-
- Size |
- {{ humanizeSize .T.Size }} |
-
-
- Discovered on |
- {{ unixTimeToYearMonthDay .T.DiscoveredOn }} |
-
-
- Files |
- {{ .T.NFiles }} |
-
-
-
- Files
-
- {{ range .F }}{{ .Path }}{{ "\t" }}{{ humanizeSizeF .Size }}{{ "\n" }}{{ end }}