function upload(response, request){
console.log("Request handler 'upload' was called.");
var form = new formidable.IncomingForm();
console.log("About to parse");
form.parse(request, function(error, fields, files){
console.log("parse done");
fs.renameSync(files.upload.path, "/tmp/test.png");
response.writeHead(200, {"Content-type": "text/html"});
response.write("Received image:<br>");
response.write("<image src=/show />");
response.end();
});
}
上面是代码,下面是报错的信息。我其实就是在按照《nodeJs入门》那本书在做练习,不知道为啥代码敲完之后,它就报这个错。
fs.renameSync(files.upload.path, "/tmp/test.png");
^
TypeError: Cannot read property 'path' of undefined
at /Users/chenxiaochun/Documents/我的资料/test/nodeJs/requestHandlers.js:36:29
at IncomingForm.<anonymous> (/Users/chenxiaochun/Documents/我的资料/test/nodeJs/node_modules/formidable/lib/incoming_form.js:118:9)
at IncomingForm.emit (events.js:67:17)
at IncomingForm._error (/Users/chenxiaochun/Documents/我的资料/test/nodeJs/node_modules/formidable/lib/incoming_form.js:248:8)
at IncomingForm.write (/Users/chenxiaochun/Documents/我的资料/test/nodeJs/node_modules/formidable/lib/incoming_form.js:145:10)
at IncomingMessage.<anonymous> (/Users/chenxiaochun/Documents/我的资料/test/nodeJs/node_modules/formidable/lib/incoming_form.js:95:12)
at IncomingMessage.emit (events.js:67:17)
at HTTPParser.onBody (http.js:113:42)
at Socket.ondata (http.js:1410:22)
at TCP.onread (net.js:374:27)
我把完整的代码贴出来吧,分为以下几个部分。 index.js
var server = require("./server");
var router = require("./router");
var requestHandlers = require("./requestHandlers");
var handle = {};
handle["/"] = requestHandlers.start;
handle["/start"] = requestHandlers.start;
handle["/upload"] = requestHandlers.upload;
handle["/show"] = requestHandlers.show;
server.start(router.route, handle);
server.js
var http = require("http");
var url = require("url");
function start(route, handle){
function onRequest(request, response){
var postData = "";
var pathname = url.parse(request.url).pathname;
console.log("Request for " + pathname + " received.");
request.setEncoding("utf8");
route(handle, pathname, response, request);
}
http.createServer(onRequest).listen(8888);
console.log("Server has started.");
}
exports.start = start;
route.js
function route(handle, pathname, response, request){
console.log("About route a request for " + pathname);
if(typeof handle[pathname] === "function"){
return handle[pathname](response, request);
}else{
console.log("No request handle for " + pathname +".");
response.writeHead("404", {"Content-type": "text/plain"});
response.write("404 not found.");
response.end();
}
}
exports.route = route;
requestHandler.js
var querystring = require("querystring");
var fs = require("fs");
var formidable = require("formidable");
function start(response){
console.log("Request handler 'start' was called.");
var body = '<html>'+
'<head>'+
'<meta http-equiv="Content-Type" '+
'content="text/html; charset=UTF-8" />'+
'</head>'+
'<body>'+
'<form action="/upload" enctype="multipart/form-data" '+
'method="post">'+
'<input type="file" name="upload">'+
'<input type="submit" value="Upload file" />'+
'</form>'+
'</body>'+
'</html>';
response.writeHead(200, {"Content-type": "text/html"});
response.write(body);
response.end();
}
function upload(response, request){
console.log("Request handler 'upload' was called.");
var form = new formidable.IncomingForm();
console.log("About to parse");
form.parse(request, function(error, fields, files){
console.log("parse done");
console.log(files);
fs.renameSync(files.upload.path, "/tmp/test.png");
response.writeHead(200, {"Content-type": "text/html"});
response.write("Received image:<br>");
response.write("<image src='/show' />");
response.end();
});
}
function show(response, postData){
console.log("Request handler 'show' was called.");
fs.readFile("/tmp/test.png", "binary", function(error, file){
if(error){
response.writeHead(500, {"Content-type": "text/plain"});
response.write(error + "\n");
response.end();
}else{
response.writeHead(200, {"Content-type": "text/plain"});
response.write(file, "binary");
response.end();
}
});
}
exports.start = start;
exports.upload = upload;
exports.show = show;
Cannot read property ‘path’ of undefined 無法讀取 undefined 的 path 屬性,也就是說 files.upload 是undefined。
是,我现在就是不明白它为啥是undefined?
查看你的request,然后看files,再看你页面上file标签的name是不是对
function start(response){
console.log("Request handler 'start' was called.");
var body = '<html>'+
'<head>'+
'<meta http-equiv="Content-Type" '+
'content="text/html; charset=UTF-8" />'+
'</head>'+
'<body>'+
'<form action="/upload" enctype="multipart/form-data" '+
'method="post">'+
'<input type="file" name="upload">'+
'<input type="submit" value="Upload file" />'+
'</form>'+
'</body>'+
'</html>';
response.writeHead(200, {"Content-type": "text/html"});
response.write(body);
response.end();
}
您说的是这里吗,应该没问题啊。
你在upload方法里,看下req里有没有files属性,有的话用 var file = req.files.upload;即可取到文件 还有,你使用的express还是connect?
这是加上您说的request.files.upload之后的upload方法:
function upload(response, request){
console.log("Request handler 'upload' was called.");
var form = new formidable.IncomingForm();
console.log("About to parse");
**console.log(request.files.upload);**
form.parse(request, function(error, fields, files){
console.log("parse done");
fs.renameSync(files.upload.path, "/tmp/test.png");
response.writeHead(200, {"Content-type": "text/html"});
response.write("Received image:<br>");
response.write("<image src='/show' />");
response.end();
});
}
可是,运行之后找不到upload属性,下面是报错信息。
console.log(request.files.upload);
^
TypeError: Cannot read property 'upload' of undefined
at Object.upload [as /upload] (/Users/chenxiaochun/Documents/我的资料/test/nodeJs/requestHandlers.js:35:27)
at route (/Users/chenxiaochun/Documents/我的资料/test/nodeJs/router.js:5:25)
at Server.onRequest (/Users/chenxiaochun/Documents/我的资料/test/nodeJs/server.js:13:3)
at Server.emit (events.js:70:17)
at HTTPParser.onIncoming (http.js:1514:12)
at HTTPParser.onHeadersComplete (http.js:102:31)
at Socket.ondata (http.js:1410:22)
at TCP.onread (net.js:374:27)
我是一个初学者,还不会使用您提到的那两个框架。
@chenxiaochun 呃,这样说好累,你直接用的formidable吧,我试下先
@sumory 呃,是好累,可我也没啥好办法。不过,还是非常感谢哈。。。
这是测试代码:
var formidable = require('formidable');
var http = require('http');
var util = require('util');
http.createServer(function(req, res) {
if (req.url == '/upload' && req.method.toLowerCase() == 'post') {
// parse a file upload
var form = new formidable.IncomingForm();
form.uploadDir = '/home/tmp';//手动设置默认上传tmp目录,可以通过fs.rename更改
form.parse(req, function(err, fields, files) {
res.writeHead(200, {'content-type': 'text/plain'});
res.write('received upload:\n\n');
res.end(util.inspect({fields: fields, files: files}));
});
return;
}
// show a file upload form
res.writeHead(200, {'content-type': 'text/html'});
res.end(
'<form action="/upload" enctype="multipart/form-data" method="post">'+
'<input type="text" name="title"><br>'+
'<input type="file" name="upload" multiple="multiple"><br>'+
'<input type="submit" value="Upload">'+
'</form>'
);
}).listen(8080);
然后我上传一个文件后的打印信息是:
received upload:
{ fields: { title: '' },
files:
{ upload:
{ size: 4037,
path: '/home/tmp/0af01b60359775ed9e02c9384cc93dcc',
name: 'status.xsd',
type: 'application/octet-stream',
hash: false,
lastModifiedDate: Thu Aug 09 2012 13:46:21 GMT+0800 (CST),
_writeStream: [Object],
length: [Getter],
filename: [Getter],
mime: [Getter] } } }
想要的信息都有了,你可以通过files.upload取得所有你想要的信息
- 非常感谢您能抽时间给我写这个实例,不过,类似您的这个实例我也已经实现了。
- 我现在是初学者,不仅仅是想实现这个文件上传的功能,其实我更想知道:我上面自己所写的代码,到底是哪里不对,为什么不对。
- 再一次感谢您的帮助。
好吧,用你写的方法,完整代码在此:
var formidable = require('formidable');
var http = require('http');
var util = require('util');
var fs = require('fs');
function upload(response, request){
console.log("Request handler 'upload' was called.");
var form = new formidable.IncomingForm();
console.log("About to parse");
form.parse(request, function(error, fields, files){
console.log("parse done");
fs.renameSync(files.upload.path, "/home/tmp/test.png");
console.log(files);
response.writeHead(200, {"Content-type": "text/html"});
response.write("Received image:<br>");
response.write("<image src=/show />");
response.end();
});
}
function start(response){
console.log("Request handler 'start' was called.");
var body = '<html>'+
'<head>'+
'<meta http-equiv="Content-Type" '+
'content="text/html; charset=UTF-8" />'+
'</head>'+
'<body>'+
'<form action="/upload" enctype="multipart/form-data" '+
'method="post">'+
'<input type="file" name="upload">'+
'<input type="submit" value="Upload file" />'+
'</form>'+
'</body>'+
'</html>';
response.writeHead(200, {"Content-type": "text/html"});
response.write(body);
response.end();
}
http.createServer(function(req, res) {
if (req.url == '/upload' && req.method.toLowerCase() == 'post') {
// parse a file upload
upload(res, req);
/*
var form = new formidable.IncomingForm();
form.uploadDir = '/home/tmp';//手动设置默认上传tmp目录,可以通过fs.rename更改
form.parse(req, function(err, fields, files) {
res.writeHead(200, {'content-type': 'text/plain'});
res.write('received upload:\n\n');
res.end(util.inspect({fields: fields, files: files}));
});
return;
*/
}
start(res);
/*
// show a file upload form
res.writeHead(200, {'content-type': 'text/html'});
res.end(
'<form action="/upload" enctype="multipart/form-data" method="post">'+
'<input type="text" name="title"><br>'+
'<input type="file" name="upload"><br>'+
'<input type="submit" value="Upload">'+
'</form>'
);
*/
}).listen(8080);
这个例子我也试过了,开始我是在win7上面,也是这样报错,我就有点怀疑是不是中文的原因,然后我开了虚拟机,在Ubuntu里面装了个,整个路径都是英文名,然后顺利运行了。楼主你也试试看看是不是这个原因
我用的是mac,我已经把整个路径都改成了英文,还是报那个错误。-_-!
@chenxiaochun 你们还真不幸。。。
@sumory 初学,很有可能就是一个低级的小错误。
因为你的 function(response,request) 是错误的,而应该是 function(request,response) .
@chenxiaochun console.log(files) 看看
form.parse(request, function(error, fields, files){
.....
}
我发现,这个位置,办理出您写的例子files对象,可以看到以下这堆东西:
{ upload:
{ size: 63949,
path: '/tmp/1189fcfbfe44620782b17efb53d23c04',
name: '1.png',
type: 'image/png',
hash: false,
lastModifiedDate: Fri, 10 Aug 2012 02:48:08 GMT,
_writeStream:
{ path: '/tmp/1189fcfbfe44620782b17efb53d23c04',
fd: 9,
writable: false,
flags: 'w',
encoding: 'binary',
mode: 438,
bytesWritten: 63949,
busy: false,
_queue: [],
drainable: true },
length: [Getter],
filename: [Getter],
mime: [Getter] } }
而我写的例子中的files对象仅仅是一个空的{},因此就造成无法找到path。
@chenxiaochun 不是吧,我不是最后给出了用‘你写的代码’的例子
@chenxiaochun 提交表单的时候,你有没有选择一个文件啊?
@leizongmin -_-!,当然选择了,我选择一个图片,然后点击上传,浏览器给我的返回结果是无法连接到localhost:8888
-
仔细看一下,它代码是这样调用的:
upload(res, req);
因此不可能是这个错误; -
若真弄反了,出错信息应该是
Cannot read property 'files' of undefined
而不是Cannot read property 'path' of undefined
@chenxiaochun 刚测试过,没有选择文件,直接点“提交”按钮确实会提示这个出错。
@leizongmin 那如果选择了文件呢?能上传成功,并且显示出来吗?
@sumory 是,您写的例子确实是用了一部分“我的代码”,而且运行起来也是正确的,我一直在拿你写的这段代码和我的代码进行比对,但我实在是看不出来,到底是哪里错了。-_-!!!
@leizongmin 是,我也测试过了,确实不是这个地方的原因。
@guilin 输出了files,它仅仅是一个空的{},不知道为什么。
@chenxiaochun 额滴神那,肯定是有地方错了呗,看看分号啊,参数顺序啊,监听端口啊,formidable版本啊,文件编码啊等等等等。。。
@sumory 哈哈,是。您说的这些地方 ,我也一直在检查,这不是看不出问题所在嘛。如果我的水平让您抓狂了,实在是报歉。
@chenxiaochun 这倒没有。实在不行先放放呗,这段代码都很简单,先用正确的,不用纠结了
Request handler 'upload' was called.
About to parse
Request handler 'start' was called.
parse done
{ upload:
{ size: 71040,
path: '\\tmp\\d8f69e1c65b5db83acdb1a109603884f',
name: 'php.ini',
type: 'application/octet-stream',
lastModifiedDate: Fri Aug 10 2012 11:47:00 GMT+0800 (中国标准时间),
_writeStream:
{ path: '\\tmp\\d8f69e1c65b5db83acdb1a109603884f',
fd: 3,
writable: false,
flags: 'w',
encoding: 'binary',
mode: 438,
bytesWritten: 71040,
busy: false,
_queue: [],
_open: [Function],
drainable: true },
length: [Getter],
filename: [Getter],
mime: [Getter] } }
Request handler 'start' was called.
@sumory 是,你说的有道理。但是心里堵着个疙瘩,实在是不爽。
@chenxiaochun 所以你贴这个代码没有用的,问题不在于这里,而在于files在哪里生成的。
@chenxiaochun 您贴个完整的代码出来看看吧。。。
@chenxiaochun 还有一个可能是你提交表单的HTML代码可能不正确,表单属性必须是enctype="multipart/form-data"才可以上传文件的
@leizongmin 我已经检查了,是这个类型的。
@leizongmin 我已经在上面把完整的源码贴出来了,麻烦您帮忙看一下。
@leizongmin 这是我写的实例的运行结果吗,竟然可以上传成功。
问题解决了:
-
在文件server.js中,你不应该用
request.setEncoding("utf8");
来手动设置字符编码,当你设置了这个之后,会导致formidable解析出错; -
在文件requestHandler.js中,
form.parse(request, function(error, fields, files){
这里没有判断是否有出错信息error
,因为若error不为null,则fileds和files就可能不能得到正确的值; -
这也解释了为什么 @sumory 给出的精简代码能正确运行,而你这一陀代码却会会出错。
这个问题搞了这么久的主要原因就是上面的第二点,以后记住,回调里面记得首先要判断是否有出错信息,若没有判断,后面会引发的一堆莫名奇妙的错误
@leizongmin 老雷牛bi。。。
刚才闲得蛋疼,上楼主所说的《Node入门》(http://www.nodebeginner.org/index-zh-cn.html )看了一下,人家的代码是这样子的:
var http = require("http");
var url = require("url");
function start(route, handle) {
function onRequest(request, response) {
var pathname = url.parse(request.url).pathname;
console.log("Request for " + pathname + " received.");
route(handle, pathname, response, request);
}
http.createServer(onRequest).listen(8888);
console.log("Server has started.");
}
exports.start = start;
本以为是《Node入门》误人子弟,原来那句request.setEncoding("utf8");
还是楼主自己加上去的,真是
一行代码引发的悲剧
@leizongmin 两眼热泪啊,感激之情,无以言表。果然是犯了一个愚蠢的错误。-_-!!!
挽楼主尊严哇。
哈哈,,为了debug,可以没有尊严。不过,各位大牛也木有不给尊严啊。
我也是因为马虎啊~~,不过错误地方不一样。 ’<input type=“file” name=“upload” multiple=“multiple”>’(修改后) 提交表单的HTML代码的问题。在win7下完美运行。谢谢各位大牛~~。撒花撒花
遇到了同样的问题,我的原因是没有写 enctype=“multipart/form-data”