zoukankan      html  css  js  c++  java
  • 使用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^_^

  • 相关阅读:
    poj 3617 Best Cow Line
    POJ 1852 Ants
    Pairs
    codility MinAbsSum
    Codeforces Beta Round #67 (Div. 2)C. Modified GCD
    timus 1018. Binary Apple Tree
    C
    HDU 1299Diophantus of Alexandria
    BZOJ2155(?) R集合 (卡特兰数)
    CSP模拟赛 number (二分+数位DP)
  • 原文地址:https://www.cnblogs.com/chuanzi/p/10514223.html
Copyright © 2011-2022 走看看