mirror of
				https://github.com/node-red/node-red.git
				synced 2025-03-01 10:36:34 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			275 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			275 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
|  * Copyright JS Foundation and other contributors, http://js.foundation
 | |
|  *
 | |
|  * Licensed under the Apache License, Version 2.0 (the "License");
 | |
|  * you may not use this file except in compliance with the License.
 | |
|  * You may obtain a copy of the License at
 | |
|  *
 | |
|  * http://www.apache.org/licenses/LICENSE-2.0
 | |
|  *
 | |
|  * Unless required by applicable law or agreed to in writing, software
 | |
|  * distributed under the License is distributed on an "AS IS" BASIS,
 | |
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
|  * See the License for the specific language governing permissions and
 | |
|  * limitations under the License.
 | |
|  **/
 | |
| 
 | |
| module.exports = function(RED) {
 | |
|     "use strict";
 | |
|     var fs = require("fs-extra");
 | |
|     var os = require("os");
 | |
|     var path = require("path");
 | |
| 
 | |
|     function FileNode(n) {
 | |
|         RED.nodes.createNode(this,n);
 | |
|         this.filename = n.filename;
 | |
|         this.appendNewline = n.appendNewline;
 | |
|         this.overwriteFile = n.overwriteFile.toString();
 | |
|         this.createDir = n.createDir || false;
 | |
|         var node = this;
 | |
|         node.wstream = null;
 | |
|         node.data = [];
 | |
| 
 | |
|         this.on("input",function(msg) {
 | |
|             var filename = node.filename || msg.filename || "";
 | |
|             if ((!node.filename) && (!node.tout)) {
 | |
|                 node.tout = setTimeout(function() {
 | |
|                     node.status({fill:"grey",shape:"dot",text:filename});
 | |
|                     clearTimeout(node.tout);
 | |
|                     node.tout = null;
 | |
|                 },333);
 | |
|             }
 | |
|             if (filename === "") {
 | |
|                 node.warn(RED._("file.errors.nofilename"));
 | |
|             } else if (node.overwriteFile === "delete") {
 | |
|                 fs.unlink(filename, function (err) {
 | |
|                     if (err) {
 | |
|                         node.error(RED._("file.errors.deletefail",{error:err.toString()}),msg);
 | |
|                     } else {
 | |
|                         if (RED.settings.verbose) {
 | |
|                             node.log(RED._("file.status.deletedfile",{file:filename}));
 | |
|                         }
 | |
|                         node.send(msg);
 | |
|                     }
 | |
|                 });
 | |
|             } else if (msg.hasOwnProperty("payload") && (typeof msg.payload !== "undefined")) {
 | |
|                 var dir = path.dirname(filename);
 | |
|                 if (node.createDir) {
 | |
|                     try {
 | |
|                         fs.ensureDirSync(dir);
 | |
|                     } catch(err) {
 | |
|                         node.error(RED._("file.errors.createfail",{error:err.toString()}),msg);
 | |
|                         return;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 var data = msg.payload;
 | |
|                 if ((typeof data === "object") && (!Buffer.isBuffer(data))) {
 | |
|                     data = JSON.stringify(data);
 | |
|                 }
 | |
|                 if (typeof data === "boolean") { data = data.toString(); }
 | |
|                 if (typeof data === "number") { data = data.toString(); }
 | |
|                 if ((node.appendNewline) && (!Buffer.isBuffer(data))) { data += os.EOL; }
 | |
|                 node.data.push({msg:msg,data:Buffer.from(data)});
 | |
| 
 | |
|                 while (node.data.length > 0) {
 | |
|                     if (node.overwriteFile === "true") {
 | |
|                         (function(packet) {
 | |
|                             node.wstream = fs.createWriteStream(filename, { encoding:'binary', flags:'w', autoClose:true });
 | |
|                             node.wstream.on("error", function(err) {
 | |
|                                 node.error(RED._("file.errors.writefail",{error:err.toString()}),msg);
 | |
|                             });
 | |
|                             node.wstream.on("open", function() {
 | |
|                                 node.wstream.end(packet.data, function() {
 | |
|                                     node.send(packet.msg);
 | |
|                                 });
 | |
|                             })
 | |
|                         })(node.data.shift());
 | |
|                     }
 | |
|                     else {
 | |
|                         // Append mode
 | |
|                         var recreateStream = !node.wstream || !node.filename;
 | |
|                         if (node.wstream && node.wstreamIno) {
 | |
|                             // There is already a stream open and we have the inode
 | |
|                             // of the file. Check the file hasn't been deleted
 | |
|                             // or deleted and recreated.
 | |
|                             try {
 | |
|                                 var stat = fs.statSync(filename);
 | |
|                                 // File exists - check the inode matches
 | |
|                                 if (stat.ino !== node.wstreamIno) {
 | |
|                                     // The file has been recreated. Close the current
 | |
|                                     // stream and recreate it
 | |
|                                     recreateStream = true;
 | |
|                                     node.wstream.end();
 | |
|                                     delete node.wstream;
 | |
|                                     delete node.wstreamIno;
 | |
|                                 }
 | |
|                             } catch(err) {
 | |
|                                 // File does not exist
 | |
|                                 recreateStream = true;
 | |
|                                 node.wstream.end();
 | |
|                                 delete node.wstream;
 | |
|                                 delete node.wstreamIno;
 | |
|                             }
 | |
|                         }
 | |
|                         if (recreateStream) {
 | |
|                             node.wstream = fs.createWriteStream(filename, { encoding:'binary', flags:'a', autoClose:true });
 | |
|                             node.wstream.on("open", function(fd) {
 | |
|                                 try {
 | |
|                                     var stat = fs.statSync(filename);
 | |
|                                     node.wstreamIno = stat.ino;
 | |
|                                 } catch(err) {
 | |
|                                 }
 | |
|                             });
 | |
|                             node.wstream.on("error", function(err) {
 | |
|                                 node.error(RED._("file.errors.appendfail",{error:err.toString()}),msg);
 | |
|                             });
 | |
|                         }
 | |
|                         if (node.filename) {
 | |
|                             // Static filename - write and reuse the stream next time
 | |
|                             var packet = node.data.shift()
 | |
|                             node.wstream.write(packet.data, function() {
 | |
|                                 node.send(packet.msg);
 | |
|                             });
 | |
| 
 | |
|                         } else {
 | |
|                             // Dynamic filename - write and close the stream
 | |
|                             var packet = node.data.shift()
 | |
|                             node.wstream.end(packet.data, function() {
 | |
|                                 node.send(packet.msg);
 | |
|                             });
 | |
|                             delete node.wstream;
 | |
|                             delete node.wstreamIno;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         });
 | |
|         this.on('close', function() {
 | |
|             if (node.wstream) { node.wstream.end(); }
 | |
|             if (node.tout) { clearTimeout(node.tout); }
 | |
|             node.status({});
 | |
|         });
 | |
|     }
 | |
|     RED.nodes.registerType("file",FileNode);
 | |
| 
 | |
| 
 | |
|     function FileInNode(n) {
 | |
|         RED.nodes.createNode(this,n);
 | |
|         this.filename = n.filename;
 | |
|         this.format = n.format;
 | |
|         this.chunk = false;
 | |
|         if (n.sendError === undefined) {
 | |
|             this.sendError = true;
 | |
|         } else {
 | |
|             this.sendError = n.sendError;
 | |
|         }
 | |
|         if (this.format === "lines") { this.chunk = true; }
 | |
|         if (this.format === "stream") { this.chunk = true; }
 | |
|         var node = this;
 | |
| 
 | |
|         this.on("input",function(msg) {
 | |
|             var filename = node.filename || msg.filename || "";
 | |
|             if (!node.filename) {
 | |
|                 node.status({fill:"grey",shape:"dot",text:filename});
 | |
|             }
 | |
|             if (filename === "") {
 | |
|                 node.warn(RED._("file.errors.nofilename"));
 | |
|             }
 | |
|             else {
 | |
|                 msg.filename = filename;
 | |
|                 var lines = Buffer.from([]);
 | |
|                 var spare = "";
 | |
|                 var count = 0;
 | |
|                 var type = "buffer";
 | |
|                 var ch = "";
 | |
|                 if (node.format === "lines") {
 | |
|                     ch = "\n";
 | |
|                     type = "string";
 | |
|                 }
 | |
|                 var hwm;
 | |
|                 var getout = false;
 | |
| 
 | |
|                 var rs = fs.createReadStream(filename)
 | |
|                     .on('readable', function () {
 | |
|                         var chunk;
 | |
|                         var hwm = rs._readableState.highWaterMark;
 | |
|                         while (null !== (chunk = rs.read())) {
 | |
|                             if (node.chunk === true) {
 | |
|                                 getout = true;
 | |
|                                 if (node.format === "lines") {
 | |
|                                     spare += chunk.toString();
 | |
|                                     var bits = spare.split("\n");
 | |
|                                     for (var i=0; i < bits.length - 1; i++) {
 | |
|                                         var m = {
 | |
|                                             payload:bits[i],
 | |
|                                             topic:msg.topic,
 | |
|                                             filename:msg.filename,
 | |
|                                             parts:{index:count, ch:ch, type:type, id:msg._msgid}
 | |
|                                         }
 | |
|                                         count += 1;
 | |
|                                         node.send(m);
 | |
|                                     }
 | |
|                                     spare = bits[i];
 | |
|                                 }
 | |
|                                 if (node.format === "stream") {
 | |
|                                     var m = {
 | |
|                                         payload:chunk,
 | |
|                                         topic:msg.topic,
 | |
|                                         filename:msg.filename,
 | |
|                                         parts:{index:count, ch:ch, type:type, id:msg._msgid}
 | |
|                                     }
 | |
|                                     count += 1;
 | |
|                                     if (chunk.length < hwm) { // last chunk is smaller that high water mark = eof
 | |
|                                         getout = false;
 | |
|                                         m.parts.count = count;
 | |
|                                     }
 | |
|                                     node.send(m);
 | |
|                                 }
 | |
|                             }
 | |
|                             else {
 | |
|                                 lines = Buffer.concat([lines,chunk]);
 | |
|                             }
 | |
|                         }
 | |
|                     })
 | |
|                     .on('error', function(err) {
 | |
|                         node.error(err, msg);
 | |
|                         if (node.sendError) {
 | |
|                             var sendMessage = RED.util.cloneMessage(msg);
 | |
|                             delete sendMessage.payload;
 | |
|                             sendMessage.error = err;
 | |
|                             node.send(sendMessage);
 | |
|                         }
 | |
|                     })
 | |
|                     .on('end', function() {
 | |
|                         if (node.chunk === false) {
 | |
|                             if (node.format === "utf8") { msg.payload = lines.toString(); }
 | |
|                             else { msg.payload = lines; }
 | |
|                             node.send(msg);
 | |
|                         }
 | |
|                         else if (node.format === "lines") {
 | |
|                             var m = { payload: spare,
 | |
|                                       parts: {
 | |
|                                           index: count,
 | |
|                                           count: count+1,
 | |
|                                           ch: ch,
 | |
|                                           type: type,
 | |
|                                           id: msg._msgid
 | |
|                                       }
 | |
|                                     };
 | |
|                             node.send(m);
 | |
|                         }
 | |
|                         else if (getout) { // last chunk same size as high water mark - have to send empty extra packet.
 | |
|                             var m = { parts:{index:count, count:count, ch:ch, type:type, id:msg._msgid} };
 | |
|                             node.send(m);
 | |
|                         }
 | |
|                     });
 | |
|             }
 | |
|         });
 | |
|         this.on('close', function() {
 | |
|             node.status({});
 | |
|         });
 | |
|     }
 | |
|     RED.nodes.registerType("file in",FileInNode);
 | |
| }
 |