node.js实际应用【模拟Apache服务器】【express】

  1. js文件
// nodemon .\src\apache_express.js

var path = require('path')
var express = require('express')
var fs = require('fs');
// var url = require('url');
var multer = require('multer');

// 文件名称后缀会丢失
// var upload = multer({dest:'upload/'})
// 解决文件名称后缀会丢失
var storage = multer.diskStorage({
  //设置上传后文件路径,uploads文件夹需要手动创建!!!
     destination: function (req, file, cb) {
         cb(null, './public/upload')
    }, 
  //给上传文件重命名,获取添加后缀名
   filename: function (req, file, cb) {
      //  var fileFormat = (file.originalname).split(".");
      //  cb(null, file.fieldname + '-' + Date.now() + "." + fileFormat[fileFormat.length - 1]);
      cb(null, file.originalname);
   }
});  
//添加配置文件到muler对象。
var upload = multer({
    storage: storage
});


var app = express()
const port = 3000

// view engine setup
app.engine('art', require('express-art-template'));
app.set('views', path.join(__dirname, 'views'));
app.set('view engine', 'art');

//static
app.use(express.static('./public/'))

// function
var getFiles =  function(res,url,callback){
  // 异步读取
  //文件夹处理
  if(!isFile(url)){
    fs.readdir(url, function(error, files){
      if (error) {
        return console.log(error) 
      }
      var dirList = [];
      var fileList =[];
  
      files.forEach(function(iterm){
        var fullPath =  path.join(url,iterm);
        // 同步读取
        var stat = fs.lstatSync(fullPath);
        if (stat.isFile()){
          fileList.push({'fullPath':fullPath,'fileSise':stat.size,'mtime':stat.mtime});
        }else{
          // console.log('iterm:',iterm)
          // console.log('fullPath:',fullPath)
          dirList.push(fullPath);
        }
      });
  
    // readdir是异步的,但是这里和forEach是同步的,也就是forEach执行完才执行这里
    // console.log('dir: ',dirList)
    // console.log('file: ',fileList)
    obj = {
      title: url,
      files: files,
      dirList:dirList,
      fileList:fileList,}
  
    callback(res,obj)
    });
    // 下面的打印会报参数未定义的错误,因为readdir异步还没执行
    // console.log('dir: ',dirList)
    // console.log('file: ',fileList)
  // 文件处理
  }else{
    fs.readFile(url, function(error, data){
      if (error) {
        return response.end('404 Not Found');
      }
      var fileEnd = path.extname(url);
      if (fileEnd === '.PNG') {
        res.setHeader('Content-Type', 'image/jpeg');
      }
      if (fileEnd === '.txt') {
        res.setHeader('Content-Type', 'text/plain; charset=utf-8');
      }
      res.send(data)
    });
  }
}


//返回渲染后页面
var render_res = function(res,obj){
  res.render('index.art',obj);
} 


//判断url对应的是否是文件
var isFile = function(url){
  var fileEnd = path.extname(url);
  return fileEnd !== ''
}

// routes
app.get('/', function (req, res) {
  getFiles(res,'.',render_res)
});


// 只能匹配一级目录
// app.get('/:url', function (req, res) {
//     var url = req.params.url
//     console.log(url)
//     getFiles(res,url,render_res)
// });


// 匹配所有目录
app.get('/*', function (req, res) {
  var url = '.'+req.url
  // console.log(url)
  getFiles(res,url,render_res)
});


// 上传
app.post('/upload', upload.single('myfile'), function(req, res, next){
  //console.log(req);
  res.send('upload OK!');
});


app.listen(port, () => {
// app.listen(port, function()  {
  console.log(`Example app listening at http://localhost:${port}`)
})



  1. index.art

<html dir="ltr" lang="zh" i18n-processed="">

<head>
  <meta charset="utf-8">
  <meta name="google" value="notranslate">
  
  <style>
    h1 {
      border-bottom: 1px solid #c0c0c0;
      margin-bottom: 10px;
      padding-bottom: 10px;
      white-space: nowrap;
    }

    table {
      border-collapse: collapse;
    }

    th {
      cursor: pointer;
    }

    td.detailsColumn {
      -webkit-padding-start: 2em;
      text-align: end;
      white-space: nowrap;
    }

    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;
    }

    html[dir=rtl] a {
      background-position-x: right;
    }

    #parentDirLinkBox {
      margin-bottom: 10px;
      padding-bottom: 10px;
    }

    #listingParsingErrorBox {
      border: 1px solid black;
      background: #fae691;
      padding: 10px;
      display: none;
    }
  </style>
  <title id="title">{{ title }}</title>
</head>

<body>
  <h1 id="header">索引</h1>
  <div id="parentDirLinkBox">
    <a id="parentDirLink" class="icon up" href="..">
    <span id="parentDirText">[上级目录]</span>
  </a>
  </div>
  <table>
    <thead>
      <tr class="header" id="theader">
        <th onclick="javascript:sortTable(0);">名称</th>
        <th class="detailsColumn" onclick="javascript:sortTable(1);">
          大小
        </th>
        <th class="detailsColumn" onclick="javascript:sortTable(2);">
          修改日期
        </th>
      </tr>
    </thead>
    <tbody id="tbody">
      <!-- file list -->
      {{each fileList}}
      <tr>
        <td data-value="{{ $value.fullPath }}"><a class="icon file" draggable="true"  href="/{{ $value.fullPath }}" >{{ $value['fullPath'] }}</a></td>
        <td class="detailsColumn" data-value="0">{{ $value.fileSise }}</td>
        <td class="detailsColumn" data-value="{{ $value.mtime }}">{{ $value.mtime }}</td>
      </tr>
      {{/each}}
      <!-- dir list -->
      {{each dirList}}
      <tr>
        <td data-value="{{ $value }}"><a class="icon dir" href="/{{ $value }}" >{{$value}}/</a></td>
        <td class="detailsColumn" data-value="0"></td>
        <td class="detailsColumn" data-value="1509589967">2017/11/2 上午10:32:47</td>
      </tr>
      {{/each}}
    </tbody>
  </table>
  <!-- upload -->
  <hr />
  <form action="/upload" method="post" enctype="multipart/form-data">
            <input type="file" name="myfile" id="myfile" value="" />
            <input type="submit" value="上传"/>
  </form>

</body>
</html>



posted @ 2021-06-19 23:03  该显示昵称已被使用了  阅读(57)  评论(0编辑  收藏  举报