diff options
| author | Felix Hanley <felix@userspace.com.au> | 2017-06-19 02:12:03 +0000 |
|---|---|---|
| committer | Felix Hanley <felix@userspace.com.au> | 2017-06-19 02:12:03 +0000 |
| commit | 08fe757de9e738c3174a51edffcc0fb87176e3fd (patch) | |
| tree | a7505557b0c1dc87b2f55b86d218fb4748093b54 | |
| parent | afe038fb57690b20229d63b64c15b07b45d71527 (diff) | |
| download | dhtsearch-08fe757de9e738c3174a51edffcc0fb87176e3fd.tar.gz dhtsearch-08fe757de9e738c3174a51edffcc0fb87176e3fd.tar.bz2 | |
Add basic pagination
| -rw-r--r-- | http.go | 100 | ||||
| -rw-r--r-- | torrent.go | 12 |
2 files changed, 84 insertions, 28 deletions
@@ -5,8 +5,15 @@ import ( "expvar" "fmt" "net/http" + "strconv" ) +type results struct { + Page int `json:"page"` + PageSize int `json:"page_size"` + Torrents []Torrent `json:"torrents"` +} + func indexHandler(w http.ResponseWriter, r *http.Request) { w.Header().Set("Cache-Control", "public") if r.URL.Path != "/" { @@ -40,27 +47,40 @@ func statsHandler(w http.ResponseWriter, r *http.Request) { func searchHandler(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json; charset=utf-8") w.Header().Set("Cache-Control", "no-cache") + + offset := 0 + page := 1 + var err error + pStr := r.URL.Query().Get("page") + if pStr != "" { + page, err = strconv.Atoi(pStr) + if err != nil { + fmt.Printf("Failed to parse page: %q\n", err) + } + offset = (page - 1) * 50 + } + if q := r.URL.Query().Get("q"); q != "" { - torrents, err := torrentsByName(q) + torrents, err := torrentsByName(q, offset) if err != nil { w.WriteHeader(500) fmt.Printf("Error: %q\n", err) return } w.WriteHeader(200) - json.NewEncoder(w).Encode(torrents) + json.NewEncoder(w).Encode(results{Page: page, PageSize: Config.Advanced.ResultsPageSize, Torrents: torrents}) return } if tag := r.URL.Query().Get("tag"); tag != "" { - torrents, err := torrentsByTag(tag) + torrents, err := torrentsByTag(tag, offset) if err != nil { w.WriteHeader(500) fmt.Printf("Error: %q\n", err) return } w.WriteHeader(200) - json.NewEncoder(w).Encode(torrents) + json.NewEncoder(w).Encode(results{Page: page, PageSize: Config.Advanced.ResultsPageSize, Torrents: torrents}) return } @@ -82,13 +102,13 @@ var html = []byte(` a { color:#000;text-decoration:none; } .header { padding:1em;border-bottom:1px solid #555;background-color:#eee; } .search { display:flex;float:left;margin:0;padding:0; } - .search__input { font-size:1.25em;padding:2px; } + .search__input { min-width:0;font-size:1.25em;padding:2px; } .page { padding:1em; } .torrent { margin-bottom:1em; } .torrent__name { display:block;overflow:hidden; } .search__button, .torrent__magnet { height:32px;width:32px;background: url() no-repeat top left; } - .search__button { display:inline-block;border:none;margin-left:5px;margin-right:5px; } - .search__button:focus { outline:none; } + .search__button { flex-shrink:0;display:inline-block;border:none;margin-left:5px;margin-right:5px; } + *:focus { outline:none; } .search__button--active { animation-name:spin;animation-duration:3s;animation-iteration-count:infinite;-webkit-animation-name:spin;-webkit-animation-duration:3s;-webkit-animation-iteration-count:infinite; } .torrent__magnet { display:block;float:left;margin-top:4px;margin-right:.5em; } .torrent__size, .torrent__file-count, .torrent__seen, .torrent__tags { display:inline-block;padding-right:1em;white-space:nowrap; } @@ -132,17 +152,30 @@ var html = []byte(` <div id="page" class="page"> </div> <script> -var humanSize = function (bytes) { +var interval = 5000 +var showSize = function (bytes) { if (bytes === 0) { - return '0 Bytes' + return '0B' } - var sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'] + var sizes = ['B', 'KB', 'MB', 'GB', 'TB'] var sizeIndex = parseInt(Math.floor(Math.log(bytes) / Math.log(1024)), 10) if (sizeIndex >= sizes.length) { sizeIndex = sizes.length - 1 } return (bytes / Math.pow(1024, sizeIndex)).toFixed(0) + sizes[sizeIndex] } +var showDuration = function (seconds) { + var s = parseInt(seconds, 10) + var d = Math.floor(s / 86400) + var h = Math.floor((s -= (d * 86400)) / 3600) + var m = Math.floor((s -= (h * 3600)) / 60) + return d + 'd ' + h + 'h ' + m + 'm ' + (s - (m * 60)) + 's' +} +var showRate = function (o, n) { + o = o || 0 + var rate = (n - o) / (interval / 1000) + return showSize(rate) + '/s' +} var processResponse = function (resp) { if (resp.status === 200 || resp.status === 0) { return resp.json() @@ -150,8 +183,12 @@ var processResponse = function (resp) { return new Error(resp.statusText) } } -var search = function (term) { +var search = function (term, page) { + if (term.length < 3) { + return + } bEl.classList.add('search__button--active') + page = parseInt(page, 10) || 1 var query = '' var tIdx = term.indexOf('tag:') if (tIdx >= 0) { @@ -159,20 +196,24 @@ var search = function (term) { } else { query = 'q=' + term.trim() } + query += '&page=' + page fetch('/search?' + query) .then(processResponse) .then(function (data) { var pre = [ - '<p>Displaying ', data.length, ' torrents.</p>', + '<p>Displaying ', data.torrents.length, ' torrents. ', + (data.page > 1) ? '<a class="pager prev" href="#">Previous</a> ' : '', + (data.torrents.length === data.page_size) ? '<a class="pager next" href="#">Next</a>' : '', + '</p>', '<ul id="results">' ].join('') - var torrents = data.map(function (t) { + var torrents = data.torrents.map(function (t) { var magnet = 'magnet:?xt=urn:btih:' + t.infohash return [ '<li class="torrent">', '<a class="torrent__magnet" href="', magnet, '"></a>', '<a class="torrent__name" href="', magnet, '">', t.name, '</a>', - '<span class="torrent__size">Size: ', humanSize(t.size), '</span>', + '<span class="torrent__size">Size: ', showSize(t.size), '</span>', '<span class="torrent__seen">Last seen: <time datetime="', t.seen, '">', new Date(t.seen).toLocaleString(), '</time></span>', '<span class="torrent__tags">Tags: ', t.tags.map(function (g) { return '<a class="tag" href="/tags/' + g + '">' + g + '</a>' }).join(', '), @@ -183,7 +224,7 @@ var search = function (term) { return [ '<li class="files__file file">', '<span class="file__path">', f.path, '</span>', - '<span class="file__size">[', humanSize(f.size), ']</span>', + '<span class="file__size">[', showSize(f.size), ']</span>', '</li>' ].join('') }).join(''), @@ -191,7 +232,11 @@ var search = function (term) { '</li>' ].join('') }).join('') - var post = '</ul>' + var post = [ + '</ul><p>' + (data.page > 1) ? '<a class="pager prev" href="#">Previous</a> ' : '', + (data.torrents.length === data.page_size) ? '<a class="pager next" href="#">Next</a>' : '', + '</p>'].join('') pEl.innerHTML = pre + torrents + post bEl.classList.remove('search__button--active') var togglers = document.getElementsByClassName('toggler') @@ -202,6 +247,18 @@ var search = function (term) { e.target.nextElementSibling.classList.toggle('files--active') }) } + var pagers = document.getElementsByClassName('pager') + for (var i = 0; i < pagers.length; i += 1) { + var el = pagers[i] + el.addEventListener('click', function (e) { + e.preventDefault() + if (el.classList.contains('next')) { + search(term, data.page + 1) + } else { + search(term, data.page - 1) + } + }) + } }) } var pEl = document.getElementById('page') @@ -209,10 +266,7 @@ var sEl = document.getElementById('search') var bEl = document.getElementById('go') bEl.addEventListener('click', function () { var term = sEl.value - if (term.length > 2) { - console.log('Search term: ', term) - search(term) - } + search(term) }) sEl.addEventListener('keyup', function (e) { if (e.keyCode === 13) { @@ -220,6 +274,7 @@ sEl.addEventListener('keyup', function (e) { } }) var statsEl = document.getElementById('stats') +var oldStats = {} var getStats = function () { fetch('/stats') .then(processResponse) @@ -229,11 +284,12 @@ var getStats = function () { '<dt class="stats__key">', k.replace(/_/g,' '), '</dt><dd class="stats__value">', - k.indexOf('bytes') === -1 ? data[k] : humanSize(data[k]), + k.indexOf('bytes') < 0 ? k.indexOf('time') < 0 ? data[k] : showDuration(data[k]) : showSize(data[k]) + ' (' + showRate(oldStats[k], data[k]) + ')', '</dd>' ].join('') }).join('') - setTimeout(getStats, 5000) + oldStats = data + setTimeout(getStats, interval) }) } getStats() @@ -100,9 +100,9 @@ func (t *Torrent) load() (err error) { return } -func torrentsByName(query string) ([]Torrent, error) { +func torrentsByName(query string, offset int) ([]Torrent, error) { torrents := []Torrent{} - err := DB.Select(&torrents, sqlSearchTorrents, fmt.Sprintf("%%%s%%", query)) + err := DB.Select(&torrents, sqlSearchTorrents, fmt.Sprintf("%%%s%%", query), offset) if err != nil { return nil, err } @@ -114,9 +114,9 @@ func torrentsByName(query string) ([]Torrent, error) { return torrents, nil } -func torrentsByTag(tag string) ([]Torrent, error) { +func torrentsByTag(tag string, offset int) ([]Torrent, error) { torrents := []Torrent{} - err := DB.Select(&torrents, sqlTorrentsByTag, tag) + err := DB.Select(&torrents, sqlTorrentsByTag, tag, offset) if err != nil { return nil, err } @@ -159,7 +159,7 @@ const ( from torrents t where t.tsv @@ plainto_tsquery($1) order by ts_rank(tsv, plainto_tsquery($1)) desc, t.seen desc - limit 100` + limit 50 offset $2` sqlTorrentsByTag = ` select t.id, t.infohash, t.name, t.size, t.seen @@ -168,7 +168,7 @@ const ( inner join tags ta on tt.tag_id = ta.id where ta.name = $1 group by t.id order by seen desc - limit 100` + limit 50 offset $2` sqlSelectFiles = `select * from files where torrent_id = $1 |
