-
Notifications
You must be signed in to change notification settings - Fork 0
/
index.js
88 lines (82 loc) · 2.95 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
const {join} = require('path')
const parseDatUrl = require('parse-dwebx-urls')
/**
* @description
* For a given HTTP accept header, provide a list of file-extensions to try.
* @param {string | undefined} accept
* @returns {string[]}
*/
function acceptHeaderExtensions (accept) {
var exts = []
var parts = (accept || '').split(',')
if (parts.includes('text/html') || (parts.length === 1 && parts[0] === '*/*')) exts = exts.concat(['.html', '.md'])
if (parts.includes('text/css')) exts.push('.css')
if (parts.includes('image/*') || parts.includes('image/apng')) exts = exts.concat(['.png', '.jpg', '.jpeg', '.gif'])
return exts
}
/**
* @description
* For a given archive, dwebx.json, request url, and request Accept header, find the file to serve
* @param {Object} archive the dwebx archive to read from
* @param {Object|undefined} manifest the dwebx archive's dwebx.json manifest
* @param {string|Object} url the request URL (can be pre-parsed by parse-dwebx-urls)
* @param {string} acceptHeader the request Accept header
* @returns {Promise<Object>} returns the Stat object with .path added
*/
module.exports = async function (archive, manifest, url, acceptHeader) {
// parse path
var urlp = typeof url === 'string' ? parseDatUrl(url, true) : url
var filepath = decodeURIComponent(urlp.path)
if (!filepath) filepath = '/'
if (filepath.indexOf('?') !== -1) filepath = filepath.slice(0, filepath.indexOf('?')) // strip off any query params
var hasTrailingSlash = filepath.endsWith('/')
// lookup entry
var entry
const tryStat = async (path) => {
// abort if we've already found it
if (entry) return
// apply the web_root config
if (manifest && manifest.web_root && !urlp.query.disable_web_root) {
if (path) {
path = join(manifest.web_root, path)
} else {
path = manifest.web_root
}
}
// attempt lookup
try {
entry = await archive.stat(path)
entry.path = path
} catch (e) {}
}
// do lookup
if (hasTrailingSlash) {
await tryStat(filepath + 'index.html')
await tryStat(filepath + 'index.md')
await tryStat(filepath)
} else {
await tryStat(filepath)
for (let ext of acceptHeaderExtensions(acceptHeader)) {
// fallback to different requested headers
await tryStat(filepath + ext)
}
if (entry && entry.isDirectory()) {
// unexpected directory, give the .html fallback a chance
let dirEntry = entry
entry = null
await tryStat(filepath + '.html') // fallback to .html
if (dirEntry && !entry) {
// no .html fallback found, stick with directory that we found
entry = dirEntry
}
}
}
// check for a fallback page
const useFallback = Boolean(manifest && manifest.fallback_page && !urlp.query.disable_fallback_page)
if (useFallback && (!entry || entry.isDirectory())) {
let tmp = entry; entry = null
await tryStat(manifest.fallback_page)
if (!entry) entry = tmp
}
return entry
}