commit
						316f0f90f7
					
				 11 changed files with 950 additions and 0 deletions
			
			
		@ -0,0 +1,3 @@ | 
				
			|||
.DS_Store | 
				
			|||
node_modules | 
				
			|||
dist | 
				
			|||
@ -0,0 +1,131 @@ | 
				
			|||
let p = console.log | 
				
			|||
 | 
				
			|||
import { sortBy } from 'lodash' | 
				
			|||
import fzy from './utils/fzy' | 
				
			|||
import download_json_file from './utils/download' | 
				
			|||
import upload_json_file from './utils/upload' | 
				
			|||
import idb_wrapper from './utils/idb_wrapper' | 
				
			|||
import state from './state' | 
				
			|||
 | 
				
			|||
let db = new idb_wrapper 'fuzzyhome', 'links', 1 | 
				
			|||
db.open! | 
				
			|||
 | 
				
			|||
global css body | 
				
			|||
	d:flex fld:column jc:center ai:center | 
				
			|||
	m:0 w:100% h:100% bg:#20222f | 
				
			|||
	ff:Open Sans | 
				
			|||
 | 
				
			|||
tag app | 
				
			|||
 | 
				
			|||
	def mount | 
				
			|||
		$input.focus! | 
				
			|||
		state.links = await db.reload! | 
				
			|||
		state.scored_links = state.links | 
				
			|||
 | 
				
			|||
	def reload_db | 
				
			|||
		state.links = await db.reload! | 
				
			|||
		state.scored_links = fzy state.links, state.query | 
				
			|||
 | 
				
			|||
	def handle_return | 
				
			|||
		window.location.href = state.scored_links[0].link | 
				
			|||
 | 
				
			|||
	def handle_shift_return | 
				
			|||
		window.location.href = 'https://www.google.com/search?q=' + state.query | 
				
			|||
 | 
				
			|||
	def name_exists query | 
				
			|||
		for { name } in state.links | 
				
			|||
			if query.trim!.toLowerCase! === name.trim!.toLowerCase! | 
				
			|||
				return yes | 
				
			|||
		return no | 
				
			|||
 | 
				
			|||
	def handle_click_create | 
				
			|||
		let query = state.query.trim! | 
				
			|||
		return if query === '' | 
				
			|||
		split_query = query.split /\s+/ | 
				
			|||
		return if split_query.length < 2 | 
				
			|||
		let link = split_query.pop! | 
				
			|||
		let name = split_query.join(" ") | 
				
			|||
		return if name_exists name | 
				
			|||
		await db.put { name, link } | 
				
			|||
		state.query = '' | 
				
			|||
		reload_db! | 
				
			|||
 | 
				
			|||
	def handle_input | 
				
			|||
		state.scored_links = fzy state.links, state.query | 
				
			|||
 | 
				
			|||
	def handle_click_delete | 
				
			|||
		let link = state.scored_links[0] | 
				
			|||
		return unless link | 
				
			|||
		return unless window.confirm "Do you really want to delete {link..name}?" | 
				
			|||
		await db.delete link | 
				
			|||
		state.query = '' | 
				
			|||
		reload_db! | 
				
			|||
 | 
				
			|||
	def handle_click_import e | 
				
			|||
		let data = await upload_json_file e | 
				
			|||
		return unless Array.isArray(data) | 
				
			|||
		for link in data | 
				
			|||
			if name_exists link.name | 
				
			|||
				console.log "Name already exists: {link.name}" | 
				
			|||
				continue | 
				
			|||
			await db.put { name: link.name, link: link.link } | 
				
			|||
		reload_db! | 
				
			|||
 | 
				
			|||
	def handle_click_export | 
				
			|||
		download_json_file JSON.stringify(state.links) | 
				
			|||
 | 
				
			|||
	def render | 
				
			|||
		<self> | 
				
			|||
 | 
				
			|||
			css self | 
				
			|||
				d:flex fld:column jc:flex-start ai:center | 
				
			|||
				bxs:0px 0px 10px rgba(0,0,0,0.35) | 
				
			|||
				w:80% h:80% max-width:700px min-height:205px | 
				
			|||
				p:40px pt:20px box-sizing:border-box rd:10px | 
				
			|||
 | 
				
			|||
			css .buttons | 
				
			|||
				d:flex fld:row jc:space-around w:100% h:50px | 
				
			|||
 | 
				
			|||
			css button, label | 
				
			|||
				d:flex fld:column jc:center ai:center | 
				
			|||
				bg:none c:purple4 bd:none cursor:pointer fl:1 | 
				
			|||
				fs:14px font-weight:bold ff:Open Sans | 
				
			|||
 | 
				
			|||
			css $input | 
				
			|||
				bd:1px solid purple4 | 
				
			|||
				w:100% h:50px ta:center fs:20px bg:none rd:5px | 
				
			|||
				bc:purple4 outline:none c:blue3 caret-color:blue3 px:20px | 
				
			|||
				transition:background 0.5s | 
				
			|||
				@focus bg:purple4/10 | 
				
			|||
 | 
				
			|||
			css .links | 
				
			|||
				d:flex fld:column jc:flex-start | 
				
			|||
				g:20px w:100% mt:20px ofy:auto fl:1 | 
				
			|||
 | 
				
			|||
			css a | 
				
			|||
				tt:capitalize td:none fs:20px c:blue3 | 
				
			|||
 | 
				
			|||
			<.buttons> | 
				
			|||
				<button@click=handle_click_create> "CREATE" | 
				
			|||
				<button@click=handle_click_delete> "DELETE" | 
				
			|||
				<button@click=handle_click_export> "EXPORT" | 
				
			|||
				<label> | 
				
			|||
					"IMPORT" | 
				
			|||
					<input[d:none] | 
				
			|||
						@change=handle_click_import | 
				
			|||
						@click=(this.value = '') | 
				
			|||
						type="file" | 
				
			|||
					> | 
				
			|||
			<input$input | 
				
			|||
				@hotkey('mod+k').capture=$input..focus | 
				
			|||
				bind=state.query | 
				
			|||
				@hotkey('return').capture=handle_return | 
				
			|||
				@hotkey('shift+return').capture=handle_shift_return | 
				
			|||
				@hotkey('esc').capture=$input..blur | 
				
			|||
				@input=handle_input | 
				
			|||
			> | 
				
			|||
			<.links> | 
				
			|||
				for { name, link } in state.scored_links | 
				
			|||
					<a href=link> name | 
				
			|||
 | 
				
			|||
imba.mount <app> | 
				
			|||
@ -0,0 +1,12 @@ | 
				
			|||
<html lang="en"> | 
				
			|||
		<head> | 
				
			|||
				<title>fuzzyhome</title> | 
				
			|||
				<meta charset="UTF-8"> | 
				
			|||
				<meta name="viewport" content="width=device-width, initial-scale=1"> | 
				
			|||
				<link href="https://fonts.googleapis.com/css2?family=Open+Sans:wght@300&display=swap" rel="stylesheet">  | 
				
			|||
				<style src='*'></style> | 
				
			|||
		</head> | 
				
			|||
		<body> | 
				
			|||
				<script type="module" src="./client.imba"></script> | 
				
			|||
		</body> | 
				
			|||
</html> | 
				
			|||
@ -0,0 +1,5 @@ | 
				
			|||
let state = {} | 
				
			|||
state.query = '' | 
				
			|||
state.links = [] | 
				
			|||
state.scored_links = [] | 
				
			|||
export default state | 
				
			|||
@ -0,0 +1,14 @@ | 
				
			|||
def get_datetime_string | 
				
			|||
	let obj = new Date!.toString!.split(" ") | 
				
			|||
	let date = obj.slice(1, 4).join("-").toLowerCase! | 
				
			|||
	let time = obj[4].split(":").join("-") | 
				
			|||
	"{date}_{time}" | 
				
			|||
 | 
				
			|||
export default def download_json_file data | 
				
			|||
	let element = document.createElement 'a' | 
				
			|||
	element.setAttribute 'href', 'data:text/plain;charset=utf-8,' + window.encodeURIComponent(data) | 
				
			|||
	element.setAttribute 'download', "{get_datetime_string!}.json" | 
				
			|||
	element.style.display = 'none' | 
				
			|||
	document.body.appendChild element | 
				
			|||
	element.click! | 
				
			|||
	document.body.removeChild element | 
				
			|||
@ -0,0 +1,95 @@ | 
				
			|||
let SCORE_MIN = -Infinity | 
				
			|||
let SCORE_MAX = Infinity | 
				
			|||
let SCORE_GAP_LEADING = -0.005 | 
				
			|||
let SCORE_GAP_TRAILING = -0.005 | 
				
			|||
let SCORE_GAP_INNER = -0.01 | 
				
			|||
let SCORE_MATCH_CONSECUTIVE = 1.0 | 
				
			|||
let SCORE_MATCH_SLASH = 0.9 | 
				
			|||
let SCORE_MATCH_WORD = 0.8 | 
				
			|||
let SCORE_MATCH_DOT = 0.6 | 
				
			|||
 | 
				
			|||
def fzy arr, query, keyname="name" | 
				
			|||
	let needle = query.trim!.toLowerCase! | 
				
			|||
	return [] unless arr.length > 0 | 
				
			|||
	let scored = [] | 
				
			|||
	let M = new Array(100_000) | 
				
			|||
	let D = new Array(100_000) | 
				
			|||
	let B = new Array(100_000) | 
				
			|||
	for obj in arr | 
				
			|||
		continue unless obj.hasOwnProperty keyname | 
				
			|||
		let haystack = obj[keyname].trim!.toLowerCase! | 
				
			|||
		continue unless has_match needle, haystack | 
				
			|||
		obj.fzy_score = score(needle, haystack, M, D, B) | 
				
			|||
		sorted_insert obj, scored | 
				
			|||
	scored | 
				
			|||
 | 
				
			|||
def score needle, haystack, M, D, match_bonus | 
				
			|||
	let n = needle.length | 
				
			|||
	let m = haystack.length | 
				
			|||
	if n < 1 or m < 1 | 
				
			|||
		return SCORE_MIN | 
				
			|||
	if n === m | 
				
			|||
		return SCORE_MAX | 
				
			|||
	if m > 1024 | 
				
			|||
		return SCORE_MIN | 
				
			|||
	compute needle, haystack, M, D, match_bonus | 
				
			|||
	M[(n - 1)*m + (m - 1)] | 
				
			|||
 | 
				
			|||
def compute needle, haystack, M, D, match_bonus | 
				
			|||
	let n = needle.length | 
				
			|||
	let m = haystack.length | 
				
			|||
	precompute_bonus haystack, match_bonus | 
				
			|||
	for i in [0 .. n - 1] | 
				
			|||
		let prev_score = SCORE_MIN | 
				
			|||
		let gap_score = i === n - 1 ? SCORE_GAP_TRAILING : SCORE_GAP_INNER | 
				
			|||
		for j in [0 .. m - 1] | 
				
			|||
			let ij = i*m + j | 
				
			|||
			let pij = (i - 1)*m + (j - 1) | 
				
			|||
			if needle[i] === haystack[j] | 
				
			|||
				let score = SCORE_MIN | 
				
			|||
				if i === 0 | 
				
			|||
					score = (j * SCORE_GAP_LEADING) + match_bonus[j] | 
				
			|||
				elif j > 0 | 
				
			|||
					score = Math.max(M[pij] + match_bonus[j], D[pij] + SCORE_MATCH_CONSECUTIVE) | 
				
			|||
				D[ij] = score | 
				
			|||
				M[ij] = prev_score = Math.max(score, prev_score + gap_score) | 
				
			|||
			else | 
				
			|||
				D[ij] = SCORE_MIN | 
				
			|||
				M[ij] = prev_score = prev_score + gap_score | 
				
			|||
 | 
				
			|||
def precompute_bonus haystack, match_bonus | 
				
			|||
	let m = haystack.length | 
				
			|||
	let last_ch = '/' | 
				
			|||
	for i in [0 .. m - 1] | 
				
			|||
		let ch = haystack[i] | 
				
			|||
		if last_ch === '/' | 
				
			|||
			match_bonus[i] = SCORE_MATCH_SLASH | 
				
			|||
		elif last_ch === '-' || last_ch === '_' || last_ch === ' ' | 
				
			|||
			match_bonus[i] = SCORE_MATCH_WORD | 
				
			|||
		elif last_ch === '.' | 
				
			|||
			match_bonus[i] = SCORE_MATCH_DOT | 
				
			|||
		else | 
				
			|||
			match_bonus[i] = 0 | 
				
			|||
		last_ch = ch | 
				
			|||
 | 
				
			|||
def has_match needle, haystack | 
				
			|||
	let i = 0 | 
				
			|||
	let n = -1 | 
				
			|||
	let letter | 
				
			|||
	while letter = needle[i++] | 
				
			|||
		if (n = haystack.indexOf(letter, n + 1)) === -1 | 
				
			|||
			return no | 
				
			|||
	return yes | 
				
			|||
 | 
				
			|||
def sorted_insert elem, arr | 
				
			|||
	let low = 0 | 
				
			|||
	let high = arr.length | 
				
			|||
	while low < high | 
				
			|||
		let mid = (low + high) >>> 1 | 
				
			|||
		if elem.fzy_score > arr[mid].fzy_score | 
				
			|||
			high = mid | 
				
			|||
		else | 
				
			|||
			low = mid + 1 | 
				
			|||
	arr.splice(low, 0, elem) | 
				
			|||
 | 
				
			|||
export default fzy | 
				
			|||
@ -0,0 +1,95 @@ | 
				
			|||
let p = console.log | 
				
			|||
 | 
				
			|||
class idb_wrapper | 
				
			|||
 | 
				
			|||
	constructor db_name, table_name, version | 
				
			|||
		db_name = db_name | 
				
			|||
		table_name = table_name | 
				
			|||
		version = version | 
				
			|||
		openRequest = null | 
				
			|||
 | 
				
			|||
	def open | 
				
			|||
 | 
				
			|||
		openRequest = global.indexedDB.open(db_name, version) | 
				
			|||
 | 
				
			|||
		openRequest.onupgradeneeded = do |event| | 
				
			|||
			p "Upgrading from DB version {event.oldVersion} to {event.newVersion}." | 
				
			|||
			let db = openRequest.result | 
				
			|||
			switch event.oldVersion | 
				
			|||
 | 
				
			|||
				when 0 | 
				
			|||
					db.createObjectStore(table_name, { keyPath: 'id', autoIncrement: true }) | 
				
			|||
 | 
				
			|||
		openRequest.onerror = do | 
				
			|||
			p "Open db error." | 
				
			|||
 | 
				
			|||
		openRequest.onsuccess = do | 
				
			|||
			p "Open db success." | 
				
			|||
			if global.navigator.storage and global.navigator.storage.persist | 
				
			|||
				global.navigator.storage.persist!.then! do |persistent| | 
				
			|||
					p "db is persistent: {persistent}" | 
				
			|||
 | 
				
			|||
	def reload | 
				
			|||
		let store | 
				
			|||
 | 
				
			|||
		while yes | 
				
			|||
			try | 
				
			|||
				store = #get_store "readonly" | 
				
			|||
				p "Get store success." | 
				
			|||
				break | 
				
			|||
			catch | 
				
			|||
				p "Failed to get store, retrying." | 
				
			|||
				await #sleep 10 | 
				
			|||
 | 
				
			|||
		let request = store.getAll! | 
				
			|||
 | 
				
			|||
		return new Promise! do |resolve| | 
				
			|||
 | 
				
			|||
			request.onsuccess = do | 
				
			|||
				p "Load db success." | 
				
			|||
				resolve request.result | 
				
			|||
				imba.commit! | 
				
			|||
 | 
				
			|||
			request.onerror = do | 
				
			|||
				p "Load db error." | 
				
			|||
				resolve no | 
				
			|||
 | 
				
			|||
	def delete obj | 
				
			|||
		let store = #get_store! | 
				
			|||
		let request = store.delete(obj.id) | 
				
			|||
 | 
				
			|||
		return new Promise! do |resolve| | 
				
			|||
 | 
				
			|||
			request.onsuccess = do | 
				
			|||
				p "deleted link: {obj}" | 
				
			|||
				resolve no | 
				
			|||
 | 
				
			|||
			request.onerror = do | 
				
			|||
				p "Failed to delete link: {obj}" | 
				
			|||
				resolve yes | 
				
			|||
 | 
				
			|||
	def put obj | 
				
			|||
		let store = #get_store! | 
				
			|||
		let request = store.put(obj) | 
				
			|||
 | 
				
			|||
		return new Promise! do |resolve| | 
				
			|||
 | 
				
			|||
			request.onsuccess = do | 
				
			|||
				p "Successfully put link: {obj}" | 
				
			|||
				resolve request.result | 
				
			|||
 | 
				
			|||
			request.onerror = do | 
				
			|||
				p "Failed to put link: {obj}" | 
				
			|||
				resolve no | 
				
			|||
 | 
				
			|||
	def #get_store permission="readwrite" | 
				
			|||
		p permission | 
				
			|||
		let db = openRequest.result | 
				
			|||
		let transaction = db.transaction(table_name, permission) | 
				
			|||
		transaction.objectStore(table_name) | 
				
			|||
 | 
				
			|||
	def #sleep ms | 
				
			|||
		new Promise! do |resolve| | 
				
			|||
			setTimeout resolve, ms | 
				
			|||
 | 
				
			|||
export default idb_wrapper | 
				
			|||
@ -0,0 +1,14 @@ | 
				
			|||
export default def upload_json_file e | 
				
			|||
	return new Promise! do |resolve| | 
				
			|||
		let files = e.target.files | 
				
			|||
		resolve no if files.length < 1 | 
				
			|||
		let file = files[0] | 
				
			|||
		let reader = new FileReader() | 
				
			|||
		reader.onloadend = do | 
				
			|||
			try | 
				
			|||
				resolve JSON.parse(reader.result) | 
				
			|||
			catch | 
				
			|||
				resolve no | 
				
			|||
		reader.onerror = do | 
				
			|||
			resolve no | 
				
			|||
		reader.readAsText(file) | 
				
			|||
@ -0,0 +1,561 @@ | 
				
			|||
{ | 
				
			|||
	"name": "fuzzyhome", | 
				
			|||
	"requires": true, | 
				
			|||
	"lockfileVersion": 1, | 
				
			|||
	"dependencies": { | 
				
			|||
		"accepts": { | 
				
			|||
			"version": "1.3.8", | 
				
			|||
			"resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", | 
				
			|||
			"integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", | 
				
			|||
			"requires": { | 
				
			|||
				"mime-types": "~2.1.34", | 
				
			|||
				"negotiator": "0.6.3" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"anymatch": { | 
				
			|||
			"version": "3.1.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", | 
				
			|||
			"integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", | 
				
			|||
			"requires": { | 
				
			|||
				"normalize-path": "^3.0.0", | 
				
			|||
				"picomatch": "^2.0.4" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"array-flatten": { | 
				
			|||
			"version": "1.1.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", | 
				
			|||
			"integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" | 
				
			|||
		}, | 
				
			|||
		"binary-extensions": { | 
				
			|||
			"version": "2.2.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", | 
				
			|||
			"integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==" | 
				
			|||
		}, | 
				
			|||
		"body-parser": { | 
				
			|||
			"version": "1.20.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.0.tgz", | 
				
			|||
			"integrity": "sha512-DfJ+q6EPcGKZD1QWUjSpqp+Q7bDQTsQIF4zfUAtZ6qk+H/3/QRhg9CEp39ss+/T2vw0+HaidC0ecJj/DRLIaKg==", | 
				
			|||
			"requires": { | 
				
			|||
				"bytes": "3.1.2", | 
				
			|||
				"content-type": "~1.0.4", | 
				
			|||
				"debug": "2.6.9", | 
				
			|||
				"depd": "2.0.0", | 
				
			|||
				"destroy": "1.2.0", | 
				
			|||
				"http-errors": "2.0.0", | 
				
			|||
				"iconv-lite": "0.4.24", | 
				
			|||
				"on-finished": "2.4.1", | 
				
			|||
				"qs": "6.10.3", | 
				
			|||
				"raw-body": "2.5.1", | 
				
			|||
				"type-is": "~1.6.18", | 
				
			|||
				"unpipe": "1.0.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"braces": { | 
				
			|||
			"version": "3.0.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", | 
				
			|||
			"integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", | 
				
			|||
			"requires": { | 
				
			|||
				"fill-range": "^7.0.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"bytes": { | 
				
			|||
			"version": "3.1.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", | 
				
			|||
			"integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==" | 
				
			|||
		}, | 
				
			|||
		"call-bind": { | 
				
			|||
			"version": "1.0.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", | 
				
			|||
			"integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", | 
				
			|||
			"requires": { | 
				
			|||
				"function-bind": "^1.1.1", | 
				
			|||
				"get-intrinsic": "^1.0.2" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"chokidar": { | 
				
			|||
			"version": "3.5.3", | 
				
			|||
			"resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", | 
				
			|||
			"integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", | 
				
			|||
			"requires": { | 
				
			|||
				"anymatch": "~3.1.2", | 
				
			|||
				"braces": "~3.0.2", | 
				
			|||
				"fsevents": "~2.3.2", | 
				
			|||
				"glob-parent": "~5.1.2", | 
				
			|||
				"is-binary-path": "~2.1.0", | 
				
			|||
				"is-glob": "~4.0.1", | 
				
			|||
				"normalize-path": "~3.0.0", | 
				
			|||
				"readdirp": "~3.6.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"content-disposition": { | 
				
			|||
			"version": "0.5.4", | 
				
			|||
			"resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", | 
				
			|||
			"integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", | 
				
			|||
			"requires": { | 
				
			|||
				"safe-buffer": "5.2.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"content-type": { | 
				
			|||
			"version": "1.0.4", | 
				
			|||
			"resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", | 
				
			|||
			"integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" | 
				
			|||
		}, | 
				
			|||
		"cookie": { | 
				
			|||
			"version": "0.5.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz", | 
				
			|||
			"integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==" | 
				
			|||
		}, | 
				
			|||
		"cookie-signature": { | 
				
			|||
			"version": "1.0.6", | 
				
			|||
			"resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", | 
				
			|||
			"integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" | 
				
			|||
		}, | 
				
			|||
		"debug": { | 
				
			|||
			"version": "2.6.9", | 
				
			|||
			"resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", | 
				
			|||
			"integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", | 
				
			|||
			"requires": { | 
				
			|||
				"ms": "2.0.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"depd": { | 
				
			|||
			"version": "2.0.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", | 
				
			|||
			"integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" | 
				
			|||
		}, | 
				
			|||
		"destroy": { | 
				
			|||
			"version": "1.2.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", | 
				
			|||
			"integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==" | 
				
			|||
		}, | 
				
			|||
		"ee-first": { | 
				
			|||
			"version": "1.1.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", | 
				
			|||
			"integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" | 
				
			|||
		}, | 
				
			|||
		"encodeurl": { | 
				
			|||
			"version": "1.0.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", | 
				
			|||
			"integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" | 
				
			|||
		}, | 
				
			|||
		"esbuild": { | 
				
			|||
			"version": "0.9.7", | 
				
			|||
			"resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.9.7.tgz", | 
				
			|||
			"integrity": "sha512-VtUf6aQ89VTmMLKrWHYG50uByMF4JQlVysb8dmg6cOgW8JnFCipmz7p+HNBl+RR3LLCuBxFGVauAe2wfnF9bLg==" | 
				
			|||
		}, | 
				
			|||
		"escape-html": { | 
				
			|||
			"version": "1.0.3", | 
				
			|||
			"resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", | 
				
			|||
			"integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" | 
				
			|||
		}, | 
				
			|||
		"etag": { | 
				
			|||
			"version": "1.8.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", | 
				
			|||
			"integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" | 
				
			|||
		}, | 
				
			|||
		"express": { | 
				
			|||
			"version": "4.18.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/express/-/express-4.18.1.tgz", | 
				
			|||
			"integrity": "sha512-zZBcOX9TfehHQhtupq57OF8lFZ3UZi08Y97dwFCkD8p9d/d2Y3M+ykKcwaMDEL+4qyUolgBDX6AblpR3fL212Q==", | 
				
			|||
			"requires": { | 
				
			|||
				"accepts": "~1.3.8", | 
				
			|||
				"array-flatten": "1.1.1", | 
				
			|||
				"body-parser": "1.20.0", | 
				
			|||
				"content-disposition": "0.5.4", | 
				
			|||
				"content-type": "~1.0.4", | 
				
			|||
				"cookie": "0.5.0", | 
				
			|||
				"cookie-signature": "1.0.6", | 
				
			|||
				"debug": "2.6.9", | 
				
			|||
				"depd": "2.0.0", | 
				
			|||
				"encodeurl": "~1.0.2", | 
				
			|||
				"escape-html": "~1.0.3", | 
				
			|||
				"etag": "~1.8.1", | 
				
			|||
				"finalhandler": "1.2.0", | 
				
			|||
				"fresh": "0.5.2", | 
				
			|||
				"http-errors": "2.0.0", | 
				
			|||
				"merge-descriptors": "1.0.1", | 
				
			|||
				"methods": "~1.1.2", | 
				
			|||
				"on-finished": "2.4.1", | 
				
			|||
				"parseurl": "~1.3.3", | 
				
			|||
				"path-to-regexp": "0.1.7", | 
				
			|||
				"proxy-addr": "~2.0.7", | 
				
			|||
				"qs": "6.10.3", | 
				
			|||
				"range-parser": "~1.2.1", | 
				
			|||
				"safe-buffer": "5.2.1", | 
				
			|||
				"send": "0.18.0", | 
				
			|||
				"serve-static": "1.15.0", | 
				
			|||
				"setprototypeof": "1.2.0", | 
				
			|||
				"statuses": "2.0.1", | 
				
			|||
				"type-is": "~1.6.18", | 
				
			|||
				"utils-merge": "1.0.1", | 
				
			|||
				"vary": "~1.1.2" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"fill-range": { | 
				
			|||
			"version": "7.0.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", | 
				
			|||
			"integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", | 
				
			|||
			"requires": { | 
				
			|||
				"to-regex-range": "^5.0.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"finalhandler": { | 
				
			|||
			"version": "1.2.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz", | 
				
			|||
			"integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==", | 
				
			|||
			"requires": { | 
				
			|||
				"debug": "2.6.9", | 
				
			|||
				"encodeurl": "~1.0.2", | 
				
			|||
				"escape-html": "~1.0.3", | 
				
			|||
				"on-finished": "2.4.1", | 
				
			|||
				"parseurl": "~1.3.3", | 
				
			|||
				"statuses": "2.0.1", | 
				
			|||
				"unpipe": "~1.0.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"forwarded": { | 
				
			|||
			"version": "0.2.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", | 
				
			|||
			"integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==" | 
				
			|||
		}, | 
				
			|||
		"fresh": { | 
				
			|||
			"version": "0.5.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", | 
				
			|||
			"integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" | 
				
			|||
		}, | 
				
			|||
		"fsevents": { | 
				
			|||
			"version": "2.3.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", | 
				
			|||
			"integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", | 
				
			|||
			"optional": true | 
				
			|||
		}, | 
				
			|||
		"function-bind": { | 
				
			|||
			"version": "1.1.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", | 
				
			|||
			"integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" | 
				
			|||
		}, | 
				
			|||
		"get-intrinsic": { | 
				
			|||
			"version": "1.1.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.1.tgz", | 
				
			|||
			"integrity": "sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q==", | 
				
			|||
			"requires": { | 
				
			|||
				"function-bind": "^1.1.1", | 
				
			|||
				"has": "^1.0.3", | 
				
			|||
				"has-symbols": "^1.0.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"glob-parent": { | 
				
			|||
			"version": "5.1.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", | 
				
			|||
			"integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", | 
				
			|||
			"requires": { | 
				
			|||
				"is-glob": "^4.0.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"has": { | 
				
			|||
			"version": "1.0.3", | 
				
			|||
			"resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", | 
				
			|||
			"integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", | 
				
			|||
			"requires": { | 
				
			|||
				"function-bind": "^1.1.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"has-symbols": { | 
				
			|||
			"version": "1.0.3", | 
				
			|||
			"resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", | 
				
			|||
			"integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==" | 
				
			|||
		}, | 
				
			|||
		"http-errors": { | 
				
			|||
			"version": "2.0.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", | 
				
			|||
			"integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", | 
				
			|||
			"requires": { | 
				
			|||
				"depd": "2.0.0", | 
				
			|||
				"inherits": "2.0.4", | 
				
			|||
				"setprototypeof": "1.2.0", | 
				
			|||
				"statuses": "2.0.1", | 
				
			|||
				"toidentifier": "1.0.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"iconv-lite": { | 
				
			|||
			"version": "0.4.24", | 
				
			|||
			"resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", | 
				
			|||
			"integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", | 
				
			|||
			"requires": { | 
				
			|||
				"safer-buffer": ">= 2.1.2 < 3" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"imba": { | 
				
			|||
			"version": "2.0.0-alpha.207", | 
				
			|||
			"resolved": "https://registry.npmjs.org/imba/-/imba-2.0.0-alpha.207.tgz", | 
				
			|||
			"integrity": "sha512-WhDzqTkCFmL0njkMaz4u+IEAdHQp/8Wq7LzFmjeJYQjacTZzFJCW5zlNUZbD7dqbsn44iEHrPNp5QNEBFApkzA==", | 
				
			|||
			"requires": { | 
				
			|||
				"chokidar": "^3.4.3", | 
				
			|||
				"esbuild": "^0.9.7" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"inherits": { | 
				
			|||
			"version": "2.0.4", | 
				
			|||
			"resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", | 
				
			|||
			"integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" | 
				
			|||
		}, | 
				
			|||
		"ipaddr.js": { | 
				
			|||
			"version": "1.9.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", | 
				
			|||
			"integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" | 
				
			|||
		}, | 
				
			|||
		"is-binary-path": { | 
				
			|||
			"version": "2.1.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", | 
				
			|||
			"integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", | 
				
			|||
			"requires": { | 
				
			|||
				"binary-extensions": "^2.0.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"is-extglob": { | 
				
			|||
			"version": "2.1.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", | 
				
			|||
			"integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=" | 
				
			|||
		}, | 
				
			|||
		"is-glob": { | 
				
			|||
			"version": "4.0.3", | 
				
			|||
			"resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", | 
				
			|||
			"integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", | 
				
			|||
			"requires": { | 
				
			|||
				"is-extglob": "^2.1.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"is-number": { | 
				
			|||
			"version": "7.0.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", | 
				
			|||
			"integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==" | 
				
			|||
		}, | 
				
			|||
		"lodash": { | 
				
			|||
			"version": "4.17.21", | 
				
			|||
			"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", | 
				
			|||
			"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" | 
				
			|||
		}, | 
				
			|||
		"media-typer": { | 
				
			|||
			"version": "0.3.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", | 
				
			|||
			"integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" | 
				
			|||
		}, | 
				
			|||
		"merge-descriptors": { | 
				
			|||
			"version": "1.0.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", | 
				
			|||
			"integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" | 
				
			|||
		}, | 
				
			|||
		"methods": { | 
				
			|||
			"version": "1.1.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", | 
				
			|||
			"integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" | 
				
			|||
		}, | 
				
			|||
		"mime": { | 
				
			|||
			"version": "1.6.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", | 
				
			|||
			"integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" | 
				
			|||
		}, | 
				
			|||
		"mime-db": { | 
				
			|||
			"version": "1.52.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", | 
				
			|||
			"integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==" | 
				
			|||
		}, | 
				
			|||
		"mime-types": { | 
				
			|||
			"version": "2.1.35", | 
				
			|||
			"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", | 
				
			|||
			"integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", | 
				
			|||
			"requires": { | 
				
			|||
				"mime-db": "1.52.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"ms": { | 
				
			|||
			"version": "2.0.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", | 
				
			|||
			"integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" | 
				
			|||
		}, | 
				
			|||
		"negotiator": { | 
				
			|||
			"version": "0.6.3", | 
				
			|||
			"resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", | 
				
			|||
			"integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==" | 
				
			|||
		}, | 
				
			|||
		"normalize-path": { | 
				
			|||
			"version": "3.0.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", | 
				
			|||
			"integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" | 
				
			|||
		}, | 
				
			|||
		"object-inspect": { | 
				
			|||
			"version": "1.12.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.0.tgz", | 
				
			|||
			"integrity": "sha512-Ho2z80bVIvJloH+YzRmpZVQe87+qASmBUKZDWgx9cu+KDrX2ZDH/3tMy+gXbZETVGs2M8YdxObOh7XAtim9Y0g==" | 
				
			|||
		}, | 
				
			|||
		"on-finished": { | 
				
			|||
			"version": "2.4.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", | 
				
			|||
			"integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", | 
				
			|||
			"requires": { | 
				
			|||
				"ee-first": "1.1.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"parseurl": { | 
				
			|||
			"version": "1.3.3", | 
				
			|||
			"resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", | 
				
			|||
			"integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" | 
				
			|||
		}, | 
				
			|||
		"path-to-regexp": { | 
				
			|||
			"version": "0.1.7", | 
				
			|||
			"resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", | 
				
			|||
			"integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" | 
				
			|||
		}, | 
				
			|||
		"picomatch": { | 
				
			|||
			"version": "2.3.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", | 
				
			|||
			"integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==" | 
				
			|||
		}, | 
				
			|||
		"proxy-addr": { | 
				
			|||
			"version": "2.0.7", | 
				
			|||
			"resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", | 
				
			|||
			"integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", | 
				
			|||
			"requires": { | 
				
			|||
				"forwarded": "0.2.0", | 
				
			|||
				"ipaddr.js": "1.9.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"qs": { | 
				
			|||
			"version": "6.10.3", | 
				
			|||
			"resolved": "https://registry.npmjs.org/qs/-/qs-6.10.3.tgz", | 
				
			|||
			"integrity": "sha512-wr7M2E0OFRfIfJZjKGieI8lBKb7fRCH4Fv5KNPEs7gJ8jadvotdsS08PzOKR7opXhZ/Xkjtt3WF9g38drmyRqQ==", | 
				
			|||
			"requires": { | 
				
			|||
				"side-channel": "^1.0.4" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"range-parser": { | 
				
			|||
			"version": "1.2.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", | 
				
			|||
			"integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" | 
				
			|||
		}, | 
				
			|||
		"raw-body": { | 
				
			|||
			"version": "2.5.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz", | 
				
			|||
			"integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==", | 
				
			|||
			"requires": { | 
				
			|||
				"bytes": "3.1.2", | 
				
			|||
				"http-errors": "2.0.0", | 
				
			|||
				"iconv-lite": "0.4.24", | 
				
			|||
				"unpipe": "1.0.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"readdirp": { | 
				
			|||
			"version": "3.6.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", | 
				
			|||
			"integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", | 
				
			|||
			"requires": { | 
				
			|||
				"picomatch": "^2.2.1" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"safe-buffer": { | 
				
			|||
			"version": "5.2.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", | 
				
			|||
			"integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" | 
				
			|||
		}, | 
				
			|||
		"safer-buffer": { | 
				
			|||
			"version": "2.1.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", | 
				
			|||
			"integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" | 
				
			|||
		}, | 
				
			|||
		"send": { | 
				
			|||
			"version": "0.18.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz", | 
				
			|||
			"integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==", | 
				
			|||
			"requires": { | 
				
			|||
				"debug": "2.6.9", | 
				
			|||
				"depd": "2.0.0", | 
				
			|||
				"destroy": "1.2.0", | 
				
			|||
				"encodeurl": "~1.0.2", | 
				
			|||
				"escape-html": "~1.0.3", | 
				
			|||
				"etag": "~1.8.1", | 
				
			|||
				"fresh": "0.5.2", | 
				
			|||
				"http-errors": "2.0.0", | 
				
			|||
				"mime": "1.6.0", | 
				
			|||
				"ms": "2.1.3", | 
				
			|||
				"on-finished": "2.4.1", | 
				
			|||
				"range-parser": "~1.2.1", | 
				
			|||
				"statuses": "2.0.1" | 
				
			|||
			}, | 
				
			|||
			"dependencies": { | 
				
			|||
				"ms": { | 
				
			|||
					"version": "2.1.3", | 
				
			|||
					"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", | 
				
			|||
					"integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" | 
				
			|||
				} | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"serve-static": { | 
				
			|||
			"version": "1.15.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz", | 
				
			|||
			"integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==", | 
				
			|||
			"requires": { | 
				
			|||
				"encodeurl": "~1.0.2", | 
				
			|||
				"escape-html": "~1.0.3", | 
				
			|||
				"parseurl": "~1.3.3", | 
				
			|||
				"send": "0.18.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"setprototypeof": { | 
				
			|||
			"version": "1.2.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", | 
				
			|||
			"integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" | 
				
			|||
		}, | 
				
			|||
		"side-channel": { | 
				
			|||
			"version": "1.0.4", | 
				
			|||
			"resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", | 
				
			|||
			"integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", | 
				
			|||
			"requires": { | 
				
			|||
				"call-bind": "^1.0.0", | 
				
			|||
				"get-intrinsic": "^1.0.2", | 
				
			|||
				"object-inspect": "^1.9.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"statuses": { | 
				
			|||
			"version": "2.0.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", | 
				
			|||
			"integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==" | 
				
			|||
		}, | 
				
			|||
		"to-regex-range": { | 
				
			|||
			"version": "5.0.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", | 
				
			|||
			"integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", | 
				
			|||
			"requires": { | 
				
			|||
				"is-number": "^7.0.0" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"toidentifier": { | 
				
			|||
			"version": "1.0.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", | 
				
			|||
			"integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==" | 
				
			|||
		}, | 
				
			|||
		"type-is": { | 
				
			|||
			"version": "1.6.18", | 
				
			|||
			"resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", | 
				
			|||
			"integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", | 
				
			|||
			"requires": { | 
				
			|||
				"media-typer": "0.3.0", | 
				
			|||
				"mime-types": "~2.1.24" | 
				
			|||
			} | 
				
			|||
		}, | 
				
			|||
		"unpipe": { | 
				
			|||
			"version": "1.0.0", | 
				
			|||
			"resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", | 
				
			|||
			"integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" | 
				
			|||
		}, | 
				
			|||
		"utils-merge": { | 
				
			|||
			"version": "1.0.1", | 
				
			|||
			"resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", | 
				
			|||
			"integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" | 
				
			|||
		}, | 
				
			|||
		"vary": { | 
				
			|||
			"version": "1.1.2", | 
				
			|||
			"resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", | 
				
			|||
			"integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" | 
				
			|||
		} | 
				
			|||
	} | 
				
			|||
} | 
				
			|||
@ -0,0 +1,12 @@ | 
				
			|||
{ | 
				
			|||
	"name": "fuzzyhome", | 
				
			|||
	"scripts": { | 
				
			|||
		"start": "imba -w server.imba", | 
				
			|||
		"build": "imba build server.imba" | 
				
			|||
	}, | 
				
			|||
	"dependencies": { | 
				
			|||
		"express": "^4.17.1", | 
				
			|||
		"imba": "^2.0.0-alpha.207", | 
				
			|||
		"lodash": "^4.17.21" | 
				
			|||
	} | 
				
			|||
} | 
				
			|||
@ -0,0 +1,8 @@ | 
				
			|||
import express from 'express' | 
				
			|||
import index from './app/index.html' | 
				
			|||
const app = express! | 
				
			|||
app.get(/.*/) do(req,res) | 
				
			|||
	unless req.accepts(['image/*', 'html']) == 'html' | 
				
			|||
		return res.sendStatus(404) | 
				
			|||
	res.send index.body | 
				
			|||
imba.serve app.listen(process.env.PORT or 3000) | 
				
			|||
					Loading…
					
					
				
		Reference in new issue