使用nodejs实现apache的部分功能

首先来一个配置文件config.js,这个文件导出一个包含配置信息的对象

module.exports = {
    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'
    }
}

然后是服务器代码app.js,这个文件开启一个http服务,实现了apache的目录浏览和部分类型文件的查看功能

const http = require('http')
const fs = require('fs')
const path = require('path')
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 exitst = fs.existsSync(tmp)
    if(exitst){
        let stats1 = fs.statSync(tmp)

        if(stats1.isDirectory()){
            for(let key in config.directoryIndex) {
                let file = tmp + '/' + config.directoryIndex[key]
                console.log(file);
                if(fs.existsSync(file)){
                    res.writeHead(301, {'Location': 'http://127.0.0.1:' + config.port + url + config.directoryIndex[key]})
                    res.end()
                    return
                }
            }
            if(!config.directoryBrowse){
                //没有开放目录浏览权限
                res.end('403 forbidden!!')
                return
            }
            //遍历目录
            fs.readFile('./template.html', function (err, data) {
                if (err) {
                    return res.end('404 Not Found!!!')
                }
                // 1. 如何得到 documentRoot 目录列表中的文件名和目录名
                //    fs.readdir
                // 2. 如何将得到的文件名和目录名替换到 template.html 中
                //    2.1 在 template.html 中需要替换的位置预留一个特殊的标记(就像以前使用模板引擎的标记一样)
                //    2.2 根据 files 生成需要的 HTML 内容
                let prevDisplay = 'block'
                if(url == '/'){
                    prevDisplay = 'none'
                }
                fs.readdir(tmp, async function (err, files) {
                    if (err) {
                        return res.end('Can not find dir.')
                    }
                    // 2.1 生成需要替换的内容
                    let content = ''
                    files.forEach(function (item) {
                    let type = ''
                    let separate = ''
                    let stats2 = fs.statSync(tmp + item)
                    if(stats2.isDirectory()){
                        type = 'dir'
                        separate = '/'
                    }else{
                        type = 'file'
                    }
                    content += `
                        <tr>
                            <td data-value="apple/"><a class="icon ${type}" href="http://127.0.0.1:${config.port + url + item + separate}">${item + separate}</a></td>
                        </tr>
                    `
                })
                // 替换
                data = data.toString()
                data = data.replace('^_^', content)
                data = data.replace('{{display}}', `style="display:${prevDisplay}"`)
                data = data.replace(/{{parentPath}}/g, url.substring(0, url.substr(0,url.length - 1).lastIndexOf('/') + 1))
                data = data.replace(/{{path}}/g, url.substring(1))
                // 发送解析替换过后的响应数据
                res.end(data)
                })
            })
        }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(`running in port:${config.port}`)
})

最后是目录浏览的模板文件template.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">Index Of{{path}}</title>
</head>
<body>
<h1 id="header">Index Of/{{path}}</h1>
<div id="parentDirLinkBox" {{display}}>
    <a id="parentDirLink" class="icon up" href="{{parentPath}}">
        <span id="parentDirText">[上级目录]</span>
    </a>
</div>
<table>
    <tbody id="tbody">^_^</tbody>
</table>
</body>
</html>

end^_^

猜你喜欢

转载自www.cnblogs.com/chuanzi/p/10513830.html