又拍云 Node.js 实现文件上传、删除
Node.js 服务端
使用 Node.js + Express.js 实现 服务端
const express = require("express");
const app = express();
const axios = require('axios');
app.set('port', process.env.PORT || 8082);
// 静态资源目录,这里放在了根目录,生产环境不允许这样
app.use(express.static(__dirname));
// 启动一个端口为 8082 的服务器
app.listen(app.get('port'), () => {
console.log("http://localhost:" + app.get('port'));
});
准备 Base64、HMAC-SHA1、MD5 实现签名认证
const crypto = require("crypto");
// MD5
function MD5(value) {
return crypto
.createHash("md5")
.update(value)
.digest("hex");
}
// Base64
function base64(value) {
return Buffer.from(value).toString("base64");
}
// hmacsha1
function hmacsha1(secret, value) {
return crypto.createHmac('sha1', secret).update(value, 'utf-8').digest().toString('base64');
}
上传、删除接口
const date = new Date().toGMTString();
const bucketname = ""; // 空间名
const key = ""; // 操作员
const secret = ""; // 密码
const upyunUrl = 'http://v0.api.upyun.com/'
// Upload
app.get("/api/token/upload", (req, res) => {
let fileName = (Math.random() * 100000000) >>> 0;
let expiration = ((Date.now() / 1000) >>> 0) + 30 * 60; // 请求的过期时间,UNIX UTC 时间戳,单位秒。建议设为 30 分钟 http://docs.upyun.com/api/form_api/
let method = "POST";
let policy = base64(
JSON.stringify({
bucket: bucketname,
// "save-key": "/" + fileName + "{.suffix}",
"save-key": "/{filename}{.suffix}",
expiration: expiration
})
);
let authorization =
"UPYUN " +
key +
":" +
hmacsha1(MD5(secret), method + "&/" + bucketname + "&" + policy);
res.json({
msg: "OK",
code: 200,
data: {
authorization: authorization,
policy: policy
}
});
});
// Delete
app.get('/api/token/del', (req, res) => {
let item = req.query.item;
let method = "DELETE"
let authorization = "UPYUN " +
key +
":" +
hmacsha1(MD5(secret), method + '&/' + bucketname + item + '&'+ date);
axios({
url: upyunUrl + bucketname + item,
method: 'DELETE',
headers: {
'Authorization': authorization,
'Date': date
}
}).then(response => {
res.json({
msg: "OK",
code: 200,
data: {}
});
}).catch(err => {
console.log('err', err)
})
})
跨域接口调用
const cors = require('cors');
// CORS @see https://github.com/expressjs/cors
app.use(cors());
前端
前端使用 Vue.js 实现
引入 Bootstrap.css
<link rel="stylesheet" type="text/css" href="https://unpkg.com/bootstrap@4.1.3/dist/css/bootstrap.css">
<script src="https://unpkg.com/axios/dist/axios.min.js"></script>
<!-- HTML -->
<div id="app">
<div class="card" style="margin: 50px auto; width: 300px;">
<div class="card-body">
<h5 class="card-title">UPYun Upload & Delete</h5>
<div class="card-text">
<div class="form-group">
<label for="file">Upload</label>
<input type="file" id="file" class="form-control-file" @change="onChange">
<div class="form-text text-muted">
<ul>
<li v-for="(item, index) in files">
{{item}} <a href="javascript:;" @click="onDel(item, index)">Del</a>
</li>
</ul>
</div>
</div>
</div>
</div>
</div>
</div>
引入 Vue.js、Axios
<script src="https://unpkg.com/vue@2.5.17/dist/vue.js"></script>
<script src="https://unpkg.com/axios/dist/axios.min.js"></script>
JS
const upUrl = 'http://v0.api.upyun.com/' // +空间名,如:http://v0.api.upyun.com/yun-temp
const baseApi = 'http://localhost:8082/api/'
let uploadInput;
let app = new Vue({
el: '#app',
data: {
files: []
},
methods: {
onChange: function () {
getToken(token => {
let formData = new FormData();
formData.append("file", uploadInput.files[0])
formData.append('policy', token.policy)
formData.append("authorization", token.authorization)
axios({
method: 'POST',
url: upUrl,
data: formData
}).then(res => {
res = res || {}
if (res.status !== 200) {
console.log('error')
return
}
let data = res.data || {}
this.files.push(data.url)
alert('Success')
}).catch(err => {
console.log(err);
});
});
},
onDel: function (item, index) {
this.files.splice(index, 1)
axios.request({
url: baseApi + 'token/del',
method: 'GET',
params: {
item: encodeURI(item)
}
}).then(res => {
alert('Deleted.')
}).catch(err => {
console.log(err)
})
}
},
mounted () {
uploadInput = $('file')
}
})
// DOM 获取元素
function $ (el) {
return document.getElementById(el)
}
// 获取 token
function getToken (fn) {
let token = window.localStorage.getItem('token');
token = JSON.parse(token) || {};
let nowTime = Date.now();
if (nowTime < token.expired && token.authorization && token.policy) {
fn(token)
return
}
axios({
method: 'get',
url: baseApi + 'token/upload'
})
.then(res => {
let data = res.data || {}
data = data.data || {}
const authorization = data.authorization
const policy = data.policy
const expired = ((Date.now() / 1000) >>> 0) + 30 * 60;
token = {
authorization,
policy,
expired
}
fn(token)
window.localStorage.setItem('token', JSON.stringify(token))
});
}
项目源码
转载请注明出处: https://blog.givebest.cn/other/2018/10/27/upyun-upload-delete-node.js.html
又拍云 Node.js 实现文件上传、删除的更多相关文章
- Node.js——获取文件上传进度
https://juejin.im/post/5a77a46cf265da4e78327552?utm_medium=fe&utm_source=weixinqun
- node中间层实现文件上传
一般情况下,前端的文件上传一般都是通过form表单的(<input type="file" />)来完成文件的上传,如果使用node中间层完成跨域,文件的上传就需要在n ...
- js获取文件上传进度
js获取文件上传进度: <input name="file" id="FileUpload" type="file" /> &l ...
- Nodejs学习笔记(八)--- Node.js + Express 实现上传文件功能(felixge/node-formidable)
目录 前言 formidable简介 创建项目并安装formidable 实现上传功能 运行结果 部分疑惑解析 写在之后 前言 前面讲了一个构建网站的示例,这次在此基础上再说说web的常规功能---- ...
- Nodejs学习笔记(八)—Node.js + Express 实现上传文件功能(felixge/node-formidable)
前言 前面讲了一个构建网站的示例,这次在此基础上再说说web的常规功能----文件上传,示例以一个上传图片的功能为例子 上传功能命名用formidable实现,示例很简单! PS:最近比较忙,距上一次 ...
- node+express实现文件上传功能
在进行node web开发时,我们可能经常遇到上传文件的问题,这一块如果我们没有经验,可能会遇到很多坑,下面我将跟大家分享一下,实现文件上传的一些方式. 一.node+express文件上传的常用方式 ...
- 基于node.js的websocket上传小功能
一.node.js 在目录里新建index.js var ws = require("nodejs-websocket"); console.log("开始建立连接... ...
- JS大文件上传断点续传解决方案
1 背景 用户本地有一份txt或者csv文件,无论是从业务数据库导出.还是其他途径获取,当需要使用蚂蚁的大数据分析工具进行数据加工.挖掘和共创应用的时候,首先要将本地文件上传至ODPS,普通的小文件通 ...
- JS大文件上传解决方案
1 背景 用户本地有一份txt或者csv文件,无论是从业务数据库导出.还是其他途径获取,当需要使用蚂蚁的大数据分析工具进行数据加工.挖掘和共创应用的时候,首先要将本地文件上传至ODPS,普通的小文件通 ...
随机推荐
- java通过年月得到该月每一天的日期
public static List<String> getDayByMonth(int yearParam,int monthParam){ List<String> lis ...
- node.js中path路径模块的使用
path模块是node.js中处理路径的核心模块.可以很方便的处理关于文件路径的问题. join() 将多个参数值合并成一个路径 const path = require('path'); conso ...
- OSI网络七层协议+火了火了火
因为部门新进了一台价值百万的网络测试设备,所以有太大的必要了解有关网络相关的基础知识了. 网络七层协议OSI(open system interconnection)从上到下依次为:应用层.表示层.会 ...
- 【转】背后的故事之 - 快乐的Lambda表达式(一)
快乐的Lambda表达式(二) 自从Lambda随.NET Framework3.5出现在.NET开发者眼前以来,它已经给我们带来了太多的欣喜.它优雅,对开发者更友好,能提高开发效率,天啊!它还有可能 ...
- [C#.net]未在本地计算机上注册“microsoft.ACE.oledb.12.0”提供程序
C#操作Execl文件常用的2种方式是NPOI和Microsoft.Jet.OleDb,前者需要引用dll,后者需要安装Office客户端 今天发现别人电脑使用Microsoft.Jet.OleDb能 ...
- wind量化交易
https://www.joinquant.com/study?f=home&m=memu https://www.v2ex.com/member/mushroomqiu https://sa ...
- 关于wifi网络基本原理了解
对于esp32,其wifi功能还是十分强大的,为了能够良好的完成wifi的相关开发,这里需要计算机网络的结构体系进行大致的了解. 一.网络结构分层 对于计算机网络结构,大体上可以分为5层结构: 物理层 ...
- Django Model 基础
程序涉及到数据库相关操作时,一般都会这样: 创建数据库,设计表结构和字段 使用 pymysql 来连接数据库,并编写数据访问层代码 业务逻辑层去调用数据访问层执行数据库操作 import pymysq ...
- cf 20C Dijkstra?
带队列 dijkstra #include <iostream> #include <cstdio> #include <queue> #include < ...
- 剑指offer编程题Java实现——面试题8旋转数组的最小数字
剑指offer面试题8:旋转数组的最小数字 题目:把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转.输入一个递增排序数组的一个旋转,输出旋转数组的最小元素. 例如数组{3,4,5,1, ...