255 lines
5.8 KiB
JavaScript
255 lines
5.8 KiB
JavaScript
import fs from 'fs';
|
|
import path from 'path';
|
|
import express from 'express';
|
|
|
|
import args from './args.js';
|
|
import {DEBUG, say, sleep, APP_ROOT, SNIP_CONTEXT} from './common.js';
|
|
import Archivist from './archivist.js';
|
|
|
|
const SITE_PATH = path.resolve(APP_ROOT, 'public');
|
|
|
|
const app = express();
|
|
const INDEX_FILE = args.index_file;
|
|
|
|
let running = false;
|
|
let Server, upAt, port;
|
|
|
|
const LibraryServer = {
|
|
start, stop
|
|
}
|
|
|
|
export default LibraryServer;
|
|
|
|
async function start({server_port}) {
|
|
if ( running ) {
|
|
DEBUG && console.warn(`Attempting to start server when it is not closed. Exiting start()...`);
|
|
return;
|
|
}
|
|
running = true;
|
|
try {
|
|
port = server_port;
|
|
addHandlers();
|
|
Server = app.listen(Number(port), err => {
|
|
if ( err ) {
|
|
running = false;
|
|
throw err;
|
|
}
|
|
upAt = new Date;
|
|
say({server_up:{upAt,port}});
|
|
});
|
|
} catch(e) {
|
|
running = false;
|
|
DEBUG && console.error(`Error starting server`, e);
|
|
}
|
|
}
|
|
|
|
function addHandlers() {
|
|
const {chrome_port} = args;
|
|
|
|
app.use(express.urlencoded({extended:true}));
|
|
app.use(express.static(SITE_PATH));
|
|
|
|
if ( !! args.library_path() ) {
|
|
app.use("/library", express.static(args.library_path()))
|
|
}
|
|
|
|
app.get('/search(.json)?', async (req, res) => {
|
|
await Archivist.isReady();
|
|
const {query, results:resultIds, HL} = await Archivist.search(req.query.query);
|
|
const results = resultIds.map(docId => Archivist.getDetails(docId));
|
|
if ( req.path.endsWith('.json') ) {
|
|
res.end(JSON.stringify({
|
|
results, query
|
|
}, null, 2));
|
|
} else {
|
|
results.forEach(r => {
|
|
r.snippet = Archivist.findOffsets(query, r.content.slice(0,150));
|
|
});
|
|
res.end(SearchResultView({results, query, HL}));
|
|
}
|
|
});
|
|
|
|
app.get('/mode', async (req, res) => {
|
|
res.end(Archivist.getMode());
|
|
});
|
|
|
|
app.get('/archive_index.html', async (req, res) => {
|
|
Archivist.saveIndex();
|
|
const index = Archivist.getIndex();
|
|
res.end(IndexView(index));
|
|
});
|
|
|
|
app.post('/mode', async (req, res) => {
|
|
const {mode} = req.body;
|
|
await Archivist.changeMode(mode);
|
|
res.end(`Mode set to ${mode}`);
|
|
});
|
|
|
|
app.get('/base_path', async (req, res) => {
|
|
res.end(args.getBasePath());
|
|
});
|
|
|
|
app.post('/base_path', async (req, res) => {
|
|
const {base_path} = req.body;
|
|
Archivist.beforePathChanged();
|
|
const change = args.updateBasePath(base_path);
|
|
|
|
if ( change ) {
|
|
await Archivist.afterPathChanged();
|
|
Server.close(async () => {
|
|
running = false;
|
|
console.log(`Server closed.`);
|
|
console.log(`Waiting 50ms...`);
|
|
await sleep(50);
|
|
start({server_port:port});
|
|
console.log(`Server restarting.`);
|
|
});
|
|
res.end(`Base path set to ${base_path} and saved to preferences. See console for progress. Server restarting...`);
|
|
} else {
|
|
res.end(`Base path not changed.`);
|
|
}
|
|
});
|
|
}
|
|
|
|
async function stop() {
|
|
let resolve;
|
|
const pr = new Promise(res => resolve = res);
|
|
|
|
console.log(`Closing library server...`);
|
|
|
|
Server.close(() => {
|
|
console.log(`Library server closed.`);
|
|
resolve();
|
|
});
|
|
|
|
return pr;
|
|
}
|
|
|
|
function IndexView(urls) {
|
|
return `
|
|
<!DOCTYPE html>
|
|
<meta charset=utf-8>
|
|
<title>Your HTML Library</title>
|
|
<style>
|
|
:root {
|
|
font-family: sans-serif;
|
|
background: lavenderblush;
|
|
}
|
|
body {
|
|
display: table;
|
|
margin: 0 auto;
|
|
background: silver;
|
|
padding: 0.5em;
|
|
box-shadow: 0 1px 1px purple;
|
|
}
|
|
form {
|
|
}
|
|
fieldset {
|
|
border: thin solid purple;
|
|
}
|
|
button, input, output {
|
|
}
|
|
input.long {
|
|
width: 100%;
|
|
min-width: 250px;
|
|
}
|
|
output {
|
|
font-size: smaller;
|
|
color: purple;
|
|
}
|
|
h1 {
|
|
margin: 0;
|
|
}
|
|
h2 {
|
|
margin-top: 0;
|
|
}
|
|
</style>
|
|
<h1><a href=/>22120</a></h1>
|
|
<h2>Internet Offline Library</h2>
|
|
<h2>Archive Index</h2>
|
|
<form method=GET action=/search>
|
|
<fieldset>
|
|
<legend>Search your archive</legend>
|
|
<input type=search name=query placeholder="search your library">
|
|
<button>Search</button>
|
|
</fieldset>
|
|
</form>
|
|
<ul>
|
|
${
|
|
urls.map(([url,{title, id}]) => `
|
|
<li>
|
|
${DEBUG ? id + ':' : ''} <a target=_blank href=${url}>${title||url}</a>
|
|
</li>
|
|
`).join('\n')
|
|
}
|
|
</ul>
|
|
`
|
|
}
|
|
|
|
function SearchResultView({results, query, HL}) {
|
|
return `
|
|
<!DOCTYPE html>
|
|
<meta charset=utf-8>
|
|
<title>${query} - 22120 search results</title>
|
|
<style>
|
|
:root {
|
|
font-family: sans-serif;
|
|
background: lavenderblush;
|
|
}
|
|
body {
|
|
display: table;
|
|
margin: 0 auto;
|
|
background: silver;
|
|
padding: 0.5em;
|
|
box-shadow: 0 1px 1px purple;
|
|
}
|
|
form {
|
|
}
|
|
fieldset {
|
|
border: thin solid purple;
|
|
}
|
|
button, input, output {
|
|
}
|
|
input.long {
|
|
width: 100%;
|
|
min-width: 250px;
|
|
}
|
|
output {
|
|
font-size: smaller;
|
|
color: purple;
|
|
}
|
|
h1 {
|
|
margin: 0;
|
|
}
|
|
h2 {
|
|
margin-top: 0;
|
|
}
|
|
</style>
|
|
<h1><a href=/>22120</a></h1>
|
|
<h2>Search results</h2>
|
|
<form method=GET action=/search>
|
|
<fieldset>
|
|
<legend>Search again</legend>
|
|
<input type=search name=query placeholder="search your library" value="${query}">
|
|
<button>Search</button>
|
|
</fieldset>
|
|
</form>
|
|
<p>
|
|
Showing results for <b>${query}</b>
|
|
</p>
|
|
<ol>
|
|
${
|
|
results.map(({snippet, url,title,id}) => `
|
|
<li>
|
|
${DEBUG ? id + ':' : ''} <a target=_blank href=${url}>${HL.get(id)?.title||title||url}</a>
|
|
<br>
|
|
<small>${(HL.get(id)?.url||url).slice(0,128)}</small>
|
|
<p>${snippet}</p>
|
|
</li>
|
|
`).join('\n')
|
|
}
|
|
</ol>
|
|
`
|
|
}
|
|
|