使用nodejs和art-template模板引擎实现apache的部分功能

升级一下上一篇的功能

首先还是配置:config.js

module.exports = {
    host: 'http://127.0.0.1',   //主机名
    port: 3000,                 //服务器端口号
    documentRoot: 'E:/webdev',  //根目录
    directoryBrowse: true,  //是否开启目录浏览功能
    directoryIndex: [    //目录默认访问页
        'index.html',
        'index.htm',
        'deflaut.html'
    ],
    charset: 'utf-8',
    mineType: {
        image: {
            gif: 'image/gif',
            jpeg: 'image/jpeg',
            jpg: 'image/jpeg',
            png: 'image/png',
        },
        text: {
            css: 'text/css',
            htm: 'text/html',
            html: 'text/html',
            js: 'application/x-javascript',
            json: 'application/json',
            pdf: 'application/pdf',
        },
        other: 'text/plain'
    }
}

接着是服务器代码:index.js

const http = require('http')
const fs = require('fs')
const path = require('path')
const tpl = require('art-template')
const config = require('./config')
const server = http.createServer()
const documentRoot = config.documentRoot

server.on('request', function (req, res) {
    let url = req.url
    console.log(url)
    let tmp = documentRoot + url
    let exist = fs.existsSync(tmp)
    if (exist) {
        let stats1 = fs.statSync(tmp)
        if (stats1.isDirectory()) {
            for (let key in config.directoryIndex) {
                let file = tmp + '/' + config.directoryIndex[key]
                console.log(`${config.host}:${config.port + url + config.directoryIndex[key]}`);
                if (fs.existsSync(file)) {
                    res.writeHead(302, {'Location': `${config.host}:${config.port + url + config.directoryIndex[key]}`})
                    res.end()
                    return
                }
            }
            if (!config.directoryBrowse) {
                //没有开放目录浏览权限
                res.writeHead(403, {"Content-Type": "text/html"});
                res.end('<h2>403 forbidden!!</h2>')
                return
            }
    fs.readFile('./template-apache.html', function (err, data) {
        if (err) {
            return res.end('404 Not Found.')
        }
        let prevDisplay = 'block'
        if (url == '/') {
            prevDisplay = 'none'
        }
        fs.readdir(tmp, function (err, files) {
            if (err) {
                return res.end('Can not find www dir.')
            }
            let fileObjArr = []
            for(key in files){
                fileObjArr[key] = {}
                fileObjArr[key].name = files[key]
                fileObjArr[key].type = 'file'
                fileObjArr[key].separate = ''
                let stats = fs.statSync(tmp + files[key])
                if (stats.isDirectory()) {
                    fileObjArr[key].type = 'dir'
                    fileObjArr[key].separate = '/'
                }
                fileObjArr[key].href = `${config.host}:${config.port + url + files[key] + fileObjArr[key].separate}`
            }
            let parentPath = url.substring(0, url.substr(0, url.length - 1).lastIndexOf('/') + 1)
            let htmlStr = tpl.render(data.toString(), {
                title: 'Index Of' + url,
                files: fileObjArr,
                prevDisplay,
                parentPath
            })
            res.end(htmlStr)
        })
    })
        } else {
            //渲染文件
            fs.readFile(tmp, function (err, data) {
                if (err) {
                    res.end()
                }
                let ext = path.extname(tmp).substring(1)
                if (ext in config.mineType.text) {
                    res.setHeader('Content-Type', `${config.mineType.text[ext]}; charset=${config.charset}`)
                } else if (ext in config.mineType.image) {
                    res.setHeader('Content-Type', `${config.mineType.text[ext]}`)
                } else {
                    res.setHeader('Content-Type', `${config.mineType.other}; charset=${config.charset}`)
                }
                res.end(data)
            })
        }
    } else {
        res.end()
    }
})
server.listen(config.port, function () {
    console.log(`server running in: ${config.host}:${config.port}`)
    const cp = require('child_process')
    cp.exec(`start ${config.host}:${config.port}`)  //自动打开浏览器
})

最后是模板:template-apache.html

<html>
<head>
    <meta charset="utf-8">
    <style>
        h1 {
            border-bottom: 1px solid #c0c0c0;
            margin-bottom: 10px;
            padding-bottom: 10px;
            white-space: nowrap;
        }
        table {
            border-collapse: collapse;
        }
        a.icon {
            -webkit-padding-start: 1.5em;
            text-decoration: none;
        }
        a.icon:hover {
            text-decoration: underline;
        }
        a.file {
            background: url(" ") left top no-repeat;
        }
        a.dir {
            background: url(" ") left top no-repeat;
        }
        a.up {
            background: url(" ") left top no-repeat;
        }
        #parentDirLinkBox {
            margin-bottom: 10px;
            padding-bottom: 10px;
        }
    </style>
    <title id="title">{{ title }}</title>
</head>

<body>
<h1 id="header">{{ title }}</h1>
<div id="parentDirLinkBox" style="display:{{ prevDisplay }}">
    <a id="parentDirLink" class="icon up" href="{{ parentPath }}">
        <span id="parentDirText">[上级目录]</span>
    </a>
</div>
<table>
    {{each files}}
    <tr>
        <td data-value="apple/"><a class="icon {{$value.type}}" href="{{$value.href}}">{{$value.name}}{{ $value.separate}}</a></td>
    </tr>
    {{/each}}
</table>
</body>

</html>

end^_^

原文地址:https://www.cnblogs.com/chuanzi/p/10514223.html