[magneticow] fully client-side rendering + better file tree + API fixes

This commit is contained in:
Bora M. Alper 2019-05-21 13:31:51 +01:00
parent 75abc0ee02
commit 2cc4c4930a
No known key found for this signature in database
GPG Key ID: 8F1A9504E1BD114D
12 changed files with 444 additions and 180 deletions

View File

@ -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))
}
}

View File

@ -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.
*

View File

@ -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 ? "&emsp;<tt>" + fileSize(e.size) + "</tt>" : ""),
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;
}
});
});
};

View File

@ -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);
}

View File

@ -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?/</.test(a)?((b=e.createElement(d||"q")).innerHTML=a,b.children):(d&&c(d)[0]||e).querySelectorAll(a):/f/.test(typeof a)?/c/.test(e.readyState)?a():c(e).on("DOMContentLoaded",a):a:k)};c.i[f="prototype"]=(c.extend=function(a){g=arguments;for(b=1;b<g.length;b++)if(f=g[b])for(d in f)a[d]=f[d];return a})(c.fn=c[f]=k,{on:function(a,d){a=a.split(h);this.map(function(c){(h[b=a[0]+(c.b$=c.b$||++p)]=h[b]||[]).push([d,a[1]]);c["add"+m](a[0],d)});return this},off:function(a,c){a=a.split(h);f="remove"+m;this.map(function(e){if(b=(g=h[a[0]+e.b$])&&g.length)for(;d=g[--b];)c&&c!=d[0]||a[1]&&a[1]!=d[1]||(e[f](a[0],d[0]),g.splice(b,1));else!a[1]&&e[f](a[0],c)});return this},is:function(a){d=(b=this[0])&&(b.matches||b["webkit"+l]||b["moz"+l]||b["ms"+l]);return!!d&&d.call(b,a)}});return c}(window,document,[],/\.(.+)/,0,"EventListener","MatchesSelector");
var create = function( tagName, props ) {
return $.extend( document.createElement( tagName ), props );
},
Tree = function( s, options ) {
var _this = this,
container = _this.container = $( s )[ 0 ],
tree = _this.tree = container.appendChild( create( 'ul', {
className: 'vtree'
}) );
_this.placeholder = options && options.placeholder;
_this._placeholder();
_this.leafs = {};
tree.addEventListener( 'click', function( evt ) {
if( $( evt.target ).is( '.vtree-leaf-label' ) ) {
_this.select( evt.target.parentNode.getAttribute('data-vtree-id') );
} else if( $( evt.target ).is( '.vtree-toggle' ) ) {
_this.toggle( evt.target.parentNode.getAttribute('data-vtree-id') );
}
});
if( options && options.contextmenu ) {
tree.addEventListener( 'contextmenu', function( evt ) {
var menu;
$( '.vtree-contextmenu' ).forEach( function( menu ) {
menu.parentNode.removeChild( menu );
});
if( $( evt.target ).is( '.vtree-leaf-label' ) ) {
evt.preventDefault();
evt.stopPropagation();
menu = create( 'menu', {
className: 'vtree-contextmenu'
});
var rect = evt.target.getBoundingClientRect();
$.extend(menu.style, {
top: (evt.target.offsetTop + rect.height).toString() + "px",
left: evt.target.offsetLeft.toString() + "px",
display: 'block'
});
options.contextmenu.forEach( function( item ) {
menu.appendChild( create( 'li', {
className: 'vtree-contextmenu-item',
innerHTML: item.label
}) ).addEventListener( 'click', item.action.bind( item, evt.target.parentNode.getAttribute('data-vtree-id') ) );
});
evt.target.parentNode.appendChild( menu );
}
});
document.addEventListener( 'click', function( evt ) {
if(evt.button === 2) return;
$( '.vtree-contextmenu' ).forEach( function( menu ) {
menu.parentNode.removeChild( menu );
});
});
}
};
Tree.prototype = {
constructor: Tree,
_dispatch: function( name, id ) {
var event;
try {
event = new CustomEvent( 'vtree-' + name, {
bubbles: true,
cancelable: true,
detail: {
id: id
}
});
} catch(e) {
event = document.createEvent( 'CustomEvent' );
event.initCustomEvent( 'vtree-' + name, true, true, { id: id });
}
( this.getLeaf( id, true ) || this.tree )
.dispatchEvent( event );
return this;
},
_placeholder: function() {
var p;
if( !this.tree.children.length && this.placeholder ) {
this.tree.innerHTML = '<li class="vtree-placeholder">' + this.placeholder + '</li>'
} 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
}));

View File

@ -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;
}

View File

@ -79,6 +79,7 @@ ul li div {
ul li div h3 {
margin: 0;
word-break: break-all;
}
a {

View File

@ -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();
}

View File

@ -3,12 +3,48 @@
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>{{ .T.Name }} - magnetico</title>
<title>Loading... - magneticow</title>
<link rel="stylesheet" href="/static/styles/reset.css">
<link rel="stylesheet" href="/static/styles/vanillatree-v0.0.3.css">
<link rel="stylesheet" href="/static/styles/essential.css">
<link rel="stylesheet" href="/static/styles/torrent.css">
<script defer src="/static/scripts/naturalSort-v0.8.1.js"></script>
<script defer src="/static/scripts/torrent.js"></script>
<script src="/static/scripts/naturalSort-v0.8.1.js"></script>
<script src="/static/scripts/mustache-v2.3.0.min.js"></script>
<script src="/static/scripts/vanillatree-v0.0.3.js"></script>
<script src="/static/scripts/common.js"></script>
<script src="/static/scripts/torrent.js"></script>
<!-- Goes into <main> -->
<script id="main-template" type="text/x-handlebars-template">
<div id="title">
<h2>{{ name }}</h2>
<a href="magnet:?xt=urn:btih:{{ infoHash }}&amp;dn={{ name }}">
<img src="/static/assets/magnet.gif" alt="Magnet link"
title="Download this torrent using magnet" />
<small>{{ infoHash }}</small>
</a>
</div>
<table>
<tr>
<th scope="row">Size</th>
<td>{{ sizeHumanised }}</td>
</tr>
<tr>
<th scope="row">Discovered on</th>
<td>{{ discoveredOnHumanised }}</td>
</tr>
<tr>
<th scope="row">Files</th>
<td>{{ nFiles }}</td>
</tr>
</table>
<h3>Better Files</h3>
<div id="fileTree"></div>
</script>
</head>
<body>
<header>
@ -18,33 +54,7 @@
</form>
</header>
<main>
<div id="title">
<h2>{{ .T.Name }}</h2>
<a href="magnet:?xt=urn:btih:{{ bytesToHex .T.InfoHash }}&amp;dn={{ .T.Name }}">
<img src="/static/assets/magnet.gif" alt="Magnet link"
title="Download this torrent using magnet" />
<small>{{ bytesToHex .T.InfoHash }}</small>
</a>
</div>
<table>
<tr>
<th scope="row">Size</th>
<td>{{ humanizeSize .T.Size }}</td>
</tr>
<tr>
<th scope="row">Discovered on</th>
<td>{{ unixTimeToYearMonthDay .T.DiscoveredOn }}</td>
</tr>
<tr>
<th scope="row">Files</th>
<td>{{ .T.NFiles }}</td>
</tr>
</table>
<h3>Files</h3>
<!-- Content of this element will be overwritten by the script -->
<pre>{{ range .F }}{{ .Path }}{{ "\t" }}{{ humanizeSizeF .Size }}{{ "\n" }}{{ end }}</pre>
</main>
</body>
</html>

View File

@ -5,13 +5,13 @@
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Search - magneticow</title>
<link rel="stylesheet" href="static/styles/reset.css">
<link rel="stylesheet" href="static/styles/essential.css">
<link rel="stylesheet" href="static/styles/torrents.css">
<link rel="stylesheet" href="/static/styles/reset.css">
<link rel="stylesheet" href="/static/styles/essential.css">
<link rel="stylesheet" href="/static/styles/torrents.css">
<script src="static/scripts/mustache-v2.3.0.min.js"></script>
<script src="static/scripts/common.js"></script>
<script src="static/scripts/torrents.js"></script>
<script src="/static/scripts/mustache-v2.3.0.min.js"></script>
<script src="/static/scripts/common.js"></script>
<script src="/static/scripts/torrents.js"></script>
<script id="item-template" type="text/x-handlebars-template">
<li>

View File

@ -1,15 +1,13 @@
package main
import (
"encoding/hex"
"fmt"
"github.com/pkg/errors"
"net/http"
"strings"
"time"
"github.com/pkg/errors"
"github.com/boramalper/magnetico/pkg/persistence"
"github.com/gorilla/mux"
)
// DONE
@ -36,39 +34,12 @@ func torrentsHandler(w http.ResponseWriter, r *http.Request) {
}
func torrentsInfohashHandler(w http.ResponseWriter, r *http.Request) {
infoHash, err := hex.DecodeString(mux.Vars(r)["infohash"])
if err != nil {
handlerError(errors.Wrap(err, "cannot decode infohash"), w)
return
}
torrent, err := database.GetTorrent(infoHash)
if err != nil {
handlerError(errors.Wrap(err, "cannot get torrent"), w)
return
}
if torrent == nil {
respondError(w, http.StatusNotFound, "torrent not found!")
return
}
files, err := database.GetFiles(infoHash)
if err != nil {
handlerError(errors.Wrap(err, "could not get files"), w)
return
}
if files == nil {
handlerError(fmt.Errorf("could not get files"), w)
return
}
_ = templates["torrent"].Execute(w, struct {
T *persistence.TorrentMetadata
F []persistence.File
}{
T: torrent,
F: files,
})
data := mustAsset("templates/torrent.html")
w.Header().Set("Content-Type", "text/html; charset=utf-8")
// Cache static resources for a day
w.Header().Set("Cache-Control", "max-age=86400")
_, _ = w.Write(data)
return
}
func statisticsHandler(w http.ResponseWriter, r *http.Request) {
@ -102,7 +73,7 @@ func feedHandler(w http.ResponseWriter, r *http.Request) {
time.Now().Unix(),
persistence.ByDiscoveredOn,
false,
N_TORRENTS,
20,
nil,
nil,
)
@ -137,6 +108,8 @@ func staticHandler(w http.ResponseWriter, r *http.Request) {
var contentType string
if strings.HasSuffix(r.URL.Path, ".css") {
contentType = "text/css; charset=utf-8"
} else if strings.HasSuffix(r.URL.Path, ".js") {
contentType = "text/javascript; charset=utf-8"
} else { // fallback option
contentType = http.DetectContentType(data)
}

View File

@ -29,8 +29,6 @@ import (
"github.com/boramalper/magnetico/pkg/persistence"
)
const N_TORRENTS = 20
var compiledOn string
// Set a Decoder instance as a package global, because it caches
@ -106,14 +104,16 @@ func main() {
router := mux.NewRouter()
router.HandleFunc("/",
BasicAuth(rootHandler, "magneticow"))
router.HandleFunc("/api/v0.1/files/{infohash:[a-f0-9]{40}}",
BasicAuth(apiFilesInfohashHandler, "magneticow"))
router.HandleFunc("/api/v0.1/statistics",
BasicAuth(apiStatisticsHandler, "magneticow"))
BasicAuth(apiStatistics, "magneticow"))
router.HandleFunc("/api/v0.1/torrents",
BasicAuth(apiTorrentsHandler, "magneticow"))
BasicAuth(apiTorrents, "magneticow"))
router.HandleFunc("/api/v0.1/torrents/{infohash:[a-f0-9]{40}}",
BasicAuth(apiTorrentsInfohashHandler, "magneticow"))
BasicAuth(apiTorrent, "magneticow"))
router.HandleFunc("/api/v0.1/torrents/{infohash:[a-f0-9]{40}}/filelist",
BasicAuth(apiFilelist, "magneticow"))
router.HandleFunc("/feed",
BasicAuth(feedHandler, "magneticow"))
router.PathPrefix("/static").HandlerFunc(
@ -169,7 +169,6 @@ func main() {
templates = make(map[string]*template.Template)
templates["feed"] = template.Must(template.New("feed").Funcs(templateFunctions).Parse(string(mustAsset("templates/feed.xml"))))
templates["homepage"] = template.Must(template.New("homepage").Funcs(templateFunctions).Parse(string(mustAsset("templates/homepage.html"))))
templates["torrent"] = template.Must(template.New("torrent").Funcs(templateFunctions).Parse(string(mustAsset("templates/torrent.html"))))
var err error
database, err = persistence.MakeDatabase(opts.Database, logger)