Разбирая вот этот туториал (автор Manuel Kiessling), наткнулся на ошибку при выполнении финального примера; описание этой ошибки и действия, которые приводят к её устранению, описаны здесь.
Собственно, вопрос - почему эти действия решают проблему?
Суть проблемы вкратце: метод parse() модуля formidable не может распарсить форму, поля fields и files остаются undefined. Казалось бы, очевидно, что проблема должна крыться где-то в определении формы, раз она не парсится, однако решение заключается в изменении кода совершенно в другом месте — убрали обработчики событий с объекта request - так что для меня причинно-следственная связь так и осталась неочевидной. Хотелось бы разобраться, что именно не давало parse() отработать правильно.
Код формы (располагается в файле requestHandlers.js):
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" multiple="multiple">' +
'<input type="submit" value="Upload File">' +
'</form>' +
'</body>' +
'</html>';
Далее эта форма парсится таким образом:
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("parsing done");
/* Возможна ошибка в Windows: попытка переименования уже существующего файла */
fs.rename(files.upload.path, "/tmp/test.png", function(err) {
if (err) {
fs.unlink("/tmp/test.png");
fs.rename(files.upload.path, "/tmp/test.png");
}
});
response.writeHead(200, {"Content-Type": "text/html"});
response.write("received image:<br/>");
response.write("<img src='/show' />");
response.end();
});
}
Решение (файл server.js):
вместо вот этих строк
// req.setEncoding("utf8");
// req.addListener("data", function(postDataChunk) {
// postData += postDataChunk;
// });
// req.addListener("end", function() {
// route(handle, pathname, res, req);
// });
оставляем просто
route(handle, pathname, res, req);
Похоже, я просто был невнимателен, и ответ на этот вопрос в туториале таки есть:
...Давайте полностью удалим всё, что касается postData в нашем сервере и обработчиках запроса — он нам не нужен для обработки загрузки файла и, мало того, — даже создает проблему: мы уже «поглотили» события data объекта request в сервере, а следовательно, form.parse, которому так же надо поглощать эти события, не сможет получить больше данных (потому что Node.js не буферизирует данные).
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;
что и требовалось.
Виртуальный выделенный сервер (VDS) становится отличным выбором
Подскажите, пожалуйста, почему выводит в консоль 1,2,3,4? В моем понимании, должны быть только 3 и 4 (первым срабатывает && где выполняется...