Permalink
Switch branches/tags
Nothing to show
Find file Copy path
Fetching contributors…
Cannot retrieve contributors at this time
414 lines (376 sloc) 15.5 KB
// Folve - A fuse filesystem that convolves audio files on-the-fly.
//
// Copyright (C) 2012 Henner Zeller <h.zeller@acm.org>
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
#include <arpa/inet.h>
#include <fcntl.h>
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <sys/types.h>
// Microhttpd's headers (at least v0.4.4) are broken and don't include stdarg.h
#include <stdarg.h>
#include <microhttpd.h>
#include <algorithm>
#include "folve-filesystem.h"
#include "status-server.h"
#include "util.h"
using folve::Appendf;
// To be used in CSS constant.
#define PROGRESS_WIDTH "300"
static const int kProgressWidth = 300;
static const size_t kMaxRetired = 20;
static const char kActiveProgress[] = "#7070ff";
static const char kRetiredProgress[] = "#d0d0d0";
static const char kSettingsUrl[] = "/settings";
// Aaah, I need to find the right Browser-Tab :)
// Sneak in a favicon without another resource access.
// TODO: make a nice icon, recognizable as something that has to do with "
// files and music ...
static const char kStartHtmlHeader[] = "<html><head>"
"<title>Folve</title>\n"
"<link rel='icon' type='image/png' "
"href='"
"AAAAAXNSR0IArs4c6QAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9wJDwUlEA/UBrsA"
"AABSSURBVCjPrZIxDgAgDAKh8f9froOTirU1ssKFYqS7Q4mktAxFRQDJcsPORMDYsDCXhn331"
"9GPwHJVuaFl3l4D1+h0UjIdbTh9SpP2KQ2AgSfVAdEQGx23tOopAAAAAElFTkSuQmCC'/>\n";
// TODO: someone with a bit more stylesheet-fu can attempt to make this
// more pretty and the HTML more compact.
// Also, maybe move to external file, makes editing much faster, then compile
// that into a C-string that we can include it in the binary.
static const char kCSS[] =
"<style type='text/css'>"
" a:link { text-decoration:none; }\n"
" a:visited { text-decoration:none; }\n"
" a:hover { text-decoration:underline; }\n"
" a:active { text-decoration:underline; }\n"
" .rounded_box, .filter_sel {\n"
" padding: 5px 15px;\n"
" border-radius: 5px;\n"
" -moz-border-radius: 5px; }\n"
" .filter_sel { font-weight:bold; }\n"
" .active { background-color:#a0a0ff; }\n"
" .inactive { background-color:#e0e0e0; }\n"
" .inactive:hover { background-color:#e0e0ff;\n"
" color: #000000;\n"
" text-decoration:none;}\n"
" .inactive:link { color: #000000;text-decoration:none;}\n"
" .inactive:visited { color: #000000;text-decoration:none;}\n"
// CSS classes used in the file-listin. Keep things compact.
" td { text-wrap:none; white-space:nowrap; }\n"
" .fn { font-size:small; text-wrap:none; white-space:nowrap; }\n" // filename
" .pf { width:" PROGRESS_WIDTH "px;\n" // progress-
" background: white; border:1px solid black; }\n" // frame
" .nf { text-align:right; }\n" // number formatting.
" .fb { background-color:#c0c0c0;"
" border-radius: 3px;\n"
" -moz-border-radius: 3px; }\n" // format box
"</style>";
// Callback function called by micro http daemon. Gets the StatusServer pointer
// in the user_argument.
int StatusServer::HandleHttp(void* user_argument,
struct MHD_Connection *connection,
const char *url, const char *method,
const char *version,
const char *upload_data, size_t *upload_size,
void**) {
StatusServer* server = (StatusServer*) user_argument;
struct MHD_Response *response;
int ret;
if (strcmp(url, kSettingsUrl) == 0) {
server->SetFilter(MHD_lookup_connection_value(connection,
MHD_GET_ARGUMENT_KIND, "f"));
server->SetDebug(MHD_lookup_connection_value(connection,
MHD_GET_ARGUMENT_KIND, "d"));
// We redirect to slash after this, to remove parameters from the GET-URL
response = MHD_create_response_from_data(0, (void*)"", MHD_NO, MHD_NO);
MHD_add_response_header(response, "Location", "/");
ret = MHD_queue_response(connection, 302, response);
} else {
const std::string &page = server->CreatePage();
response = MHD_create_response_from_data(page.length(), (void*) page.data(),
MHD_NO, MHD_NO);
MHD_add_response_header(response, "Content-Type",
"text/html; charset=utf-8");
ret = MHD_queue_response(connection, MHD_HTTP_OK, response);
}
// Tell aggressive cachers not to do so.
MHD_add_response_header(response, "Cache-Control", "no-cache");
MHD_add_response_header(response, "Expires", "24 Nov 1972 23:42:42 GMT");
MHD_destroy_response(response);
return ret;
}
StatusServer::StatusServer(FolveFilesystem *fs)
: expunged_retired_(0), total_seconds_filtered_(0),
total_seconds_music_seen_(0),
meta_refresh_time_(-1),
filesystem_(fs), daemon_(NULL), filter_switched_(false) {
fs->handler_cache()->SetObserver(this);
}
void StatusServer::SetFilter(const char *filter) {
if (filter == NULL) return;
filter_switched_ = filesystem_->SwitchCurrentConfigDir(filter);
}
void StatusServer::SetDebug(const char *dbg) {
if (!filesystem_->is_debug_ui_enabled()) return;
folve::EnableDebugLog(dbg != NULL && *dbg == '1');
}
bool StatusServer::Start(int port) {
daemon_ = MHD_start_daemon(MHD_USE_SELECT_INTERNALLY, port, NULL, NULL,
&HandleHttp, this,
MHD_OPTION_END);
return daemon_ != NULL;
}
StatusServer::~StatusServer() {
if (daemon_) MHD_stop_daemon(daemon_);
}
bool StatusServer::show_details() {
// Right now, we just make this dependent on debug output. Could be a separate
// option.
return folve::IsDebugLogEnabled();
}
// FileHandlerCache::Observer interface.
void StatusServer::RetireHandlerEvent(FileHandler *handler) {
HandlerStats stats;
handler->GetHandlerStatus(&stats); // Get last available stats.
if (stats.progress >= 0) {
total_seconds_music_seen_ += stats.duration_seconds;
total_seconds_filtered_ += stats.duration_seconds * stats.progress;
}
stats.last_access = folve::CurrentTime();
stats.status = HandlerStats::RETIRED;
folve::MutexLock l(&retired_mutex_);
retired_.push_front(stats);
while (retired_.size() > kMaxRetired) {
++expunged_retired_;
retired_.pop_back();
}
}
// As ugly #defines, so that gcc can warn about printf() format problems.
#define sMessageRowHtml \
"<td>%s</td><td colspan='3' style='font-size:small;'>%s</td>"
#define sProgressRowHtml \
"<td>%s</td>" \
"<td>%s</td>" \
"<td><div class='pf'>" \
"<div style='width:%dpx;background:%s;'>&nbsp;</div>\n</div></td>" \
"<td>%s</td>"
#define sTimeColumns \
"<td class='nf'>%2d:%02d</td><td>/</td><td class='nf'>%2d:%02d</td>"
#define sDecibelColumn \
"<td class='nf'%s>%.1f dB</td>"
void StatusServer::AppendFileInfo(const char *progress_style,
const HandlerStats &stats) {
content_.append("<tr>");
const char *status = "";
switch (stats.status) {
case HandlerStats::OPEN: status = "open"; break;
case HandlerStats::IDLE: status = "idle"; break;
case HandlerStats::RETIRED: status = "&nbsp;----&nbsp;"; break;
// no default to let the compiler detect new values.
}
if (!stats.message.empty()) {
Appendf(&content_, sMessageRowHtml, status, stats.message.c_str());
} else if (stats.progress == 0) {
Appendf(&content_, sMessageRowHtml, status, "Only header accessed");
} else {
Appendf(&content_, sProgressRowHtml, status,
stats.in_gapless ? "&rarr;" : "",
(int) (kProgressWidth * stats.progress),
progress_style,
stats.out_gapless ? "&rarr;" : "");
}
const int secs = stats.duration_seconds;
const int fract_sec = stats.progress * secs;
if (secs >= 0 && fract_sec >= 0) {
Appendf(&content_, sTimeColumns,
fract_sec / 60, fract_sec % 60,
secs / 60, secs % 60);
} else {
content_.append("<td colspan='3'>-</td>");
}
if (stats.max_output_value > 1e-6) {
Appendf(&content_, sDecibelColumn,
stats.max_output_value > 1.0 ? " style='background:#FF8080;'" : "",
20 * log10f(stats.max_output_value));
} else {
content_.append("<td>-</td>");
}
const char *filter_dir = stats.filter_dir.empty()
? "Pass Through" : stats.filter_dir.c_str();
Appendf(&content_, "<td class='fb'>&nbsp;%s (%s)&nbsp;</td>",
stats.format.c_str(), filter_dir);
Appendf(&content_,"<td class='fn'>%s</td>", stats.filename.c_str());
content_.append("</tr>\n");
}
static void AppendSanitizedUrlParam(const std::string &in, std::string *out) {
for (std::string::const_iterator i = in.begin(); i != in.end(); ++i) {
if (*i <= ' ' || *i == '\'' || (*i & 0x80)) {
Appendf(out, "%%%02x", (unsigned char) *i);
} else {
out->append(1, *i);
}
}
}
static void CreateSelection(const std::set<std::string> &options,
const std::string &selected,
std::string *result) {
if (options.size() == 1) {
result->append(selected);
return;
}
typedef std::set<std::string> Set;
for (Set::const_iterator it = options.begin(); it != options.end(); ++it) {
result->append("&nbsp;");
const bool active = (*it == selected);
const char *title = it->empty() ? "None : Pass Through" : it->c_str();
if (active) {
Appendf(result, "<span class='filter_sel active'>%s</span>\n", title);
} else {
Appendf(result, "<a class='filter_sel inactive' href='%s?f=",
kSettingsUrl);
AppendSanitizedUrlParam(*it, result);
Appendf(result, "'>%s</a>\n", title);
}
}
}
void StatusServer::AppendSettingsForm() {
content_.append("<p>Active filter: ");
std::set<std::string> available_dirs = filesystem_->GetAvailableConfigDirs();
CreateSelection(available_dirs,
filesystem_->current_config_subdir(),
&content_);
if (available_dirs.empty() == 1) {
content_.append(" (This is a boring configuration, add filter directories)");
} else if (filter_switched_) {
content_.append("&nbsp;<span class='rounded_box' "
"style='font-size:small;background:#FFFFa0;'>"
"Affects re- or newly opened files.</span>");
filter_switched_ = false; // only show once.
}
// TODO: re-add something for filesystem_->is_debug_ui_enabled()
content_.append("</p><hr/>");
}
struct CompareStats {
bool operator() (const HandlerStats *a, const HandlerStats *b) {
if (a->status < b->status) return true; // open before idle.
else if (a->status > b->status) return false;
return b->last_access < a->last_access; // reverse time.
}
};
const std::string &StatusServer::CreatePage() {
const double start = folve::CurrentTime();
// We re-use a string to avoid re-allocing memory every time we generate
// a page. Since we run with MHD_USE_SELECT_INTERNALLY, this is only accessed
// by one thread.
content_.clear();
content_.append(kStartHtmlHeader);
if (meta_refresh_time_ > 0) {
Appendf(&content_, "<meta http-equiv='refresh' content='%d'>\n",
meta_refresh_time_);
}
content_.append(kCSS);
content_.append("</head>\n");
content_.append("<body style='font-family:Sans-Serif;'>\n");
Appendf(&content_, "<center style='background-color:#A0FFA0;'>"
"Welcome to "
"<a href='https://github.com/hzeller/folve#readme'>Folve</a> "
FOLVE_VERSION "</center>\n");
if (show_details()) {
Appendf(&content_, "Convolving audio files from <code>%s</code>\n",
filesystem_->underlying_dir().c_str());
}
AppendSettingsForm();
std::vector<HandlerStats> stat_list;
filesystem_->handler_cache()->GetStats(&stat_list);
// Get statistics of active files to add to the existing ones.
double active_music_seen = 0.0;
double active_filtered = 0.0;
for (size_t i = 0; i < stat_list.size(); ++i) {
const HandlerStats &stats = stat_list[i];
if (stats.progress >= 0) {
active_music_seen += stats.duration_seconds;
active_filtered += stats.duration_seconds * stats.progress;
}
}
if (show_details()) {
const int t_seen = total_seconds_music_seen_ + active_music_seen;
const int t_filtered = total_seconds_filtered_ + active_filtered;
Appendf(&content_, "Total opening files <b>%d</b> "
".. and re-opened from recency cache <b>%d</b><br/>",
filesystem_->total_file_openings(),
filesystem_->total_file_reopen());
Appendf(&content_, "Total music seen <b>%dd %d:%02d:%02d</b> ",
t_seen / 86400, (t_seen % 86400) / 3600,
(t_seen % 3600) / 60, t_seen % 60);
Appendf(&content_, ".. and convolved <b>%dd %d:%02d:%02d</b> ",
t_filtered / 86400, (t_filtered % 86400) / 3600,
(t_filtered % 3600) / 60, t_filtered % 60);
Appendf(&content_, "(%.1f%%)<br/>",
(t_seen == 0) ? 0.0 : (100.0 * t_filtered / t_seen));
}
Appendf(&content_, "<h3>Accessed Recently</h3>\n%zd in recency cache.\n",
stat_list.size());
if (filesystem_->gapless_processing()) {
content_.append("Gapless transfers indicated with '&rarr;'.\n");
}
content_.append("<table>\n");
Appendf(&content_, "<tr><th>Stat</th><td><!--gapless in--></td>"
"<th width='%dpx'>Progress</th>" // progress bar.
"<td><!-- gapless out --></td>"
"<th>Pos</th><td></td><th>Len</th><th>Max&nbsp;out</th>"
"<th>Format&nbsp;(used&nbsp;filter)</th>"
"<th align='left'>File</th></tr>\n", kProgressWidth);
// ICE in arm 2.7.1 compiler if we sort values. So sort pointers
// to the values instead.
std::vector<HandlerStats *> stat_ptrs;
for (size_t i = 0; i < stat_list.size(); ++i) {
stat_ptrs.push_back(&stat_list[i]);
}
CompareStats comparator;
std::sort(stat_ptrs.begin(), stat_ptrs.end(), comparator);
for (size_t i = 0; i < stat_ptrs.size(); ++i) {
AppendFileInfo(kActiveProgress, *stat_ptrs[i]);
}
content_.append("</table><hr/>\n");
if (retired_.size() > 0) {
content_.append("<h3>Retired</h3>\n");
content_.append("<table>\n");
folve::MutexLock l(&retired_mutex_);
for (RetiredList::const_iterator it = retired_.begin();
it != retired_.end(); ++it) {
AppendFileInfo(kRetiredProgress, *it);
}
content_.append("</table>\n");
if (expunged_retired_ > 0) {
Appendf(&content_, "... (%d more)<p></p>", expunged_retired_);
}
content_.append("<hr/>");
}
const double duration = folve::CurrentTime() - start;
Appendf(&content_,
"<span style='float:left;font-size:x-small;'>%.2fms</span>"
"<span style='float:right;font-size:x-small;'>"
"&copy; 2012 Henner Zeller"
" | Folve is free software and comes with no warranty. "
" | Conveyed under the terms of the "
"<a href='http://www.gnu.org/licenses/gpl.html'>GPLv3</a>.</span>"
"</body></html>\n", duration * 1000.0);
return content_;
}