UNPKG

nadesiko3

Version:
297 lines (285 loc) 9.43 kB
/** 簡易HTTPサーバ */ import fs from 'fs' import http from 'http' import path from 'path' // 定数 const HTTPSERVER_LOGID = '[簡易HTTPサーバ]' const ERR_NOHTTPSERVER = '最初に『簡易HTTPサーバ起動時』を実行してサーバを起動する必要があります。' // オブジェクト type EasyURLActionType = 'static' | 'callback' type EasyURLCallback = (req: any, res: any) => void type EasyHTTPOnStart = (sys: any) => void class EasyURLItem { url: string action: EasyURLActionType path: string callback: EasyURLCallback constructor(action: EasyURLActionType) { this.action = action this.url = '' this.path = '' this.callback = (req, res) => {} } } class EasyURLDispather { server: any sys: any items: EasyURLItem[] curReq: any curRes: any isEnd: boolean usedHeader: boolean constructor(sys: any) { this.server = null this.items = [] this.sys = sys this.curReq = null this.curRes = null this.isEnd = false this.usedHeader = false } doRequest(req: any, res: any) { this.curReq = req this.curRes = res console.log(`${HTTPSERVER_LOGID} 要求あり URL=` + req.url) const params = this.parseURL(req.url) const url = params['?URL'] this.sys.__setSysVar('GETデータ', params) // URLの一致を調べてアクションを実行 const filtered = this.items.filter(v => url.startsWith(v.url)).sort((a, b) => { return b.url.length - a.url.length }) for (const it of filtered) { let isBreak = false if (it.action === 'static') { isBreak = this.doRequestStatic(req, res, it) } else if (it.action === 'callback') { isBreak = this.doRequestCallback(req, res, it) } if (isBreak) { break } } } return404(res: any) { console.error(HTTPSERVER_LOGID, 404, '見当たりません。') res.statusCode = 404 res.end('<html><meta charset="utf-8"><body><h1>404 見当たりません。</h1></body></html>') } doRequestStatic(req: any, res: any, it: EasyURLItem): boolean { let url: string = ('' + req.url).replace(/\.\./g, '') // URLの..を許可しない url = url.substring(it.url.length) let fpath = path.join(it.path, url) console.log('FILE=', fpath) if (!fs.existsSync(fpath)) { this.return404(res) return true } // ディレクトリなら index.html を確認 if (isDir(fpath)) { fpath = path.join(fpath, 'index.html') console.log('FILE(DIR)=', fpath) if (!fs.existsSync(fpath)) { this.return404(res) return true } } // ファイルを読んで返す fs.readFile(fpath, (err, data) => { if (err) { res.statusCode = 500 res.end('Failed to read file.') console.warn(HTTPSERVER_LOGID, 'read error file=', fpath) return true } const mime = getMIMEType(fpath) res.writeHead(200, { 'Content-Type': mime }) res.end(data) return true }) return true } doRequestCallback(req: any, res: any, it: EasyURLItem): boolean { this.isEnd = false this.usedHeader = false it.callback(req, res) if (!this.isEnd) { return true } return true } addItem(it: EasyURLItem) { this.items.push(it) } parseURL(uri: string): any { const params: any = {} if (uri.indexOf('?') >= 0) { const a = uri.split('?') params['?URL'] = a[0] const q = String(a[1]).split('&') for (const kv of q) { const qq = kv.split('=') const key = decodeURIComponent(qq[0]) const val = decodeURIComponent(qq[1]) params[key] = val } } else { params['?URL'] = uri } return params } } // MIMEタイプ const MimeTypes: any = { '.html': 'text/html', '.css': 'text/css', '.js': 'text/javascript', '.png': 'image/png', '.gif': 'image/gif', '.svg': 'svg+xml' } function getMIMEType(url: string) { let ext = '.txt' const m = url.match(/(\.[a-z0-9_]+)$/) if (m) { ext = m[1] } if (MimeTypes[ext]) { return MimeTypes[ext] } return 'text/plain' } // ディレクトリか判定 function isDir(pathName: string) { try { // node v12以下ではエラーがあると例外を返す const stats = fs.statSync(pathName) if (stats && stats.isDirectory()) { return true } } catch (err: any) { return false } } const PluginHttpServer = { 'meta': { type: 'const', value: { pluginName: 'plugin_httpserver', // プラグインの名前 description: 'HTTPサーバプラグイン', // プラグインの説明 pluginVersion: '3.6.0', // プラグインのバージョン nakoRuntime: ['cnako'], // 対象ランタイム nakoVersion: '3.6.0' // 要求なでしこバージョン } }, '初期化': { type: 'func', josi: [], pure: true, fn: function(sys: any) { sys.__httpserver = null } }, // @簡易HTTPサーバ 'GETデータ': { type: 'const', value: '' }, // @GETでーた '簡易HTTPサーバ起動時': { // @ポート番号PORTを指定して簡易HTTPサーバを起動して、CALLBACKを実行する。 // @かんいHTTPさーばきどうしたとき type: 'func', josi: [['を'], ['の', 'で']], pure: true, fn: function(callback: EasyHTTPOnStart, port: number, sys: any) { // 管理オブジェクトを作成する const dp = sys.__httpserver = new EasyURLDispather(sys) // サーバオブジェクトを生成 dp.server = http.createServer((req: any, res: any) => { dp.doRequest(req, res) }) // サーバ起動 dp.server.listen(port, () => { console.log(`${HTTPSERVER_LOGID} ポート番号(${port})で監視開始`) if (typeof callback === 'string') { callback = sys.__findFunc(callback) } callback(sys) }) } }, '簡易HTTPサーバ静的パス指定': { // @静的コンテンツのパスを指定。URLをPATHへマップする。 // @かんいHTTPさーばせいてきぱすしてい type: 'func', josi: [['を'], ['に', 'へ']], pure: true, fn: function(url: string, path: string, sys: any) { if (sys.__httpserver === null) { throw new Error(ERR_NOHTTPSERVER) } const dp: EasyURLDispather = sys.__httpserver const it: EasyURLItem = new EasyURLItem('static') it.url = url it.path = path dp.addItem(it) } }, '簡易HTTPサーバ受信時': { // @URLを指定して合致するリクエストが来たら処理を実行する。 // @かんいHTTPさーばじゅしんしたとき type: 'func', josi: [['を'], ['に', 'へ', 'で']], pure: true, fn: function(callback: EasyURLCallback, url: string, sys: any) { if (sys.__httpserver === null) { throw new Error(ERR_NOHTTPSERVER) } const dp: EasyURLDispather = sys.__httpserver const it: EasyURLItem = new EasyURLItem('callback') if (url === '') { url = '/' } if (url.charAt(0) !== '/') { url = '/' + url } it.url = url if (typeof callback === 'string') { callback = sys.__findFunc(callback) } it.callback = callback dp.addItem(it) } }, '簡易HTTPサーバ出力': { // @受信時に、データSを出力する。 // @かんいHTTPさーばしゅつりょく type: 'func', josi: [['を', 'と', 'の']], pure: true, fn: function(s: string, sys: any) { if (sys.__httpserver === null) { throw new Error(ERR_NOHTTPSERVER) } const dp: EasyURLDispather = sys.__httpserver if (dp.curRes === null) { throw new Error('『簡易HTTPサーバ受信時』のみ出力が可能です。') } if (!dp.usedHeader) { dp.usedHeader = true dp.curRes.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' }) } dp.curRes.end(s) dp.isEnd = true } }, '簡易HTTPサーバヘッダ出力': { // @受信時にステータスコードNOで、ヘッダHEAD(辞書形式)を出力する。 // @かんいHTTPさーばへっだしゅつりょく type: 'func', josi: [['で'], ['を', 'の', 'と']], pure: true, fn: function(no: number, head: string, sys: any) { if (sys.__httpserver === null) { throw new Error(ERR_NOHTTPSERVER) } const dp: EasyURLDispather = sys.__httpserver if (dp.curRes === null) { throw new Error('『簡易HTTPサーバ受信時』のみ出力が可能です。') } dp.curRes.writeHead(no, head) dp.isEnd = true dp.usedHeader = true } }, '簡易HTTPサーバ移動': { // @受信時にヘッダ302(リダイレクト)を出力してURLへページを移動力する。 // @かんいHTTPさーばいどう type: 'func', josi: [['へ', 'に']], pure: true, fn: function(url: string, sys: any) { if (sys.__httpserver === null) { throw new Error(ERR_NOHTTPSERVER) } const dp: EasyURLDispather = sys.__httpserver if (dp.curRes === null) { throw new Error('『簡易HTTPサーバ受信時』のみ出力が可能です。') } console.log(HTTPSERVER_LOGID, '移動=', url) dp.curRes.writeHead(302, { 'Location': url }) dp.curRes.end(`<html><body><a href="${url}">JUMP</a></body></html>`) dp.isEnd = true } } } export default PluginHttpServer